From 100bd027125bbfd7a794254fbd3d29ec294b1f0b Mon Sep 17 00:00:00 2001 From: John Plaisted Date: Thu, 3 Dec 2020 15:57:23 -0800 Subject: [PATCH] fix: turn on werror for neo4j-dao (#60) I also got rid of the tuple library while I was here, easiler than trying to get the generics right :) --- build.gradle | 26 ++------ dao-api/build.gradle | 1 - .../linkedin/metadata/dao/BaseQueryDAO.java | 44 +++++++++---- .../linkedin/metadata/dao/Neo4jQueryDAO.java | 21 +++--- .../metadata/dao/Neo4jQueryDAOTest.java | 65 +++++++++---------- 5 files changed, 77 insertions(+), 80 deletions(-) diff --git a/build.gradle b/build.gradle index 26a4a83f4..3691f29f9 100644 --- a/build.gradle +++ b/build.gradle @@ -45,7 +45,6 @@ project.ext.externalDependency = [ 'h2': 'com.h2database:h2:1.4.196', 'jacksonCore': 'com.fasterxml.jackson.core:jackson-core:2.9.7', 'jacksonDataBind': 'com.fasterxml.jackson.core:jackson-databind:2.9.7', - 'javatuples': 'org.javatuples:javatuples:1.2', 'jsonSimple': 'com.googlecode.json-simple:json-simple:1.1.1', 'junitJupiterApi': "org.junit.jupiter:junit-jupiter-api:$junitJupiterVersion", 'junitJupiterParams': "org.junit.jupiter:junit-jupiter-params:$junitJupiterVersion", @@ -66,32 +65,17 @@ project.ext.externalDependency = [ apply plugin: 'com.diffplug.spotless' apply from: "./gradle/release.gradle" -// TODO expand this to all projects and then delete this allow list. This list is letting us fix errors over time rather -// than in one big change. -def wErrorProjects = [ - project(':core-models'), - project(':dao-api'), - project(':dao-impl:ebean-dao'), - project(':dao-impl:elasticsearch-dao'), - // project(':dao-impl:neo4j-dao'), - project(':restli-resources'), - project(':testing'), - project(':validators') -] - allprojects { apply plugin: 'idea' apply plugin: 'eclipse' apply plugin: 'checkstyle' gradle.projectsEvaluated { - if (wErrorProjects.contains(project)) { - tasks.withType(JavaCompile) { - options.compilerArgs << "-Xlint:all" << "-Werror" << - "-Xlint:-deprecation" << // TODO - "-Xlint:-processing" << // TODO we have annotations like @Nonnull that need a processor - "-Xlint:-serial" // I don't think we care about having custom Exception subclasses be serializable... - } + tasks.withType(JavaCompile) { + options.compilerArgs << "-Xlint:all" << "-Werror" << + "-Xlint:-deprecation" << // TODO + "-Xlint:-processing" << // TODO we have annotations like @Nonnull that need a processor + "-Xlint:-serial" // I don't think we care about having custom Exception subclasses be serializable... } } } diff --git a/dao-api/build.gradle b/dao-api/build.gradle index 3fa8efac9..664248767 100644 --- a/dao-api/build.gradle +++ b/dao-api/build.gradle @@ -5,7 +5,6 @@ apply from: "$rootDir/gradle/java-publishing.gradle" dependencies { compile project(':core-models') compile project(':validators') - compile externalDependency.javatuples compile externalDependency.reflections compile externalDependency.commonsLang diff --git a/dao-api/src/main/java/com/linkedin/metadata/dao/BaseQueryDAO.java b/dao-api/src/main/java/com/linkedin/metadata/dao/BaseQueryDAO.java index 31b257a03..0829afed8 100644 --- a/dao-api/src/main/java/com/linkedin/metadata/dao/BaseQueryDAO.java +++ b/dao-api/src/main/java/com/linkedin/metadata/dao/BaseQueryDAO.java @@ -7,7 +7,8 @@ import java.util.List; import javax.annotation.Nonnull; import javax.annotation.Nullable; -import org.javatuples.Triplet; +import lombok.AllArgsConstructor; +import lombok.Getter; import static com.linkedin.metadata.dao.utils.QueryUtils.*; @@ -112,28 +113,47 @@ List findEntities( @Nonnull Class relationshipType, @Nonnull RelationshipFilter relationshipFilter, int minHops, int maxHops, int offset, int count); + /** + * A path to visit. + */ + @Getter + @AllArgsConstructor + public static final class TraversalPath { + /** + * Relationship type on the traverse path. + * + *

Must be a type defined in com.linkedin.metadata.relationship. + */ + private final Class relationship; + + /** + * The relationship filter to apply in a graph query. + */ + private final RelationshipFilter relationshipFilter; + + /** + * Intermediate entity type on the traverse path. + * + *

Must be a type defined in com.linkedin.metadata.entity. + */ + private final Class intermediateEntity; + } + /** * Finds a list of entities based on the given traversing paths. * - * @param sourceEntityClass the source entity class as the starting point for the query + * @param sourceEntityClass the source entity class as the starting point for the query. Must be a type defined in + * com.linkedin.metadata.entity. * @param sourceEntityFilter the filter to apply to the source entity when querying * @param traversePaths specify the traverse paths via a list of (relationship type, relationship filter, * intermediate entities) * @param count the maximum number of entities to return. Ignored if set to a non-positive value. * - * @param source ENTITY type. Starting point of the traverse path. Must be a type defined in - * com.linkedin.metadata.entity. - * @param intermediate entity type on the traverse path. Must be a type defined in - * com.linkedin.metadata.entity. - * @param relationship type on the traverse path. Must be a type defined in - * com.linkedin.metadata.relationship. * @return a list of entities that match the conditions specified in {@code filter} */ @Nonnull - public abstract - List findEntities( - @Nullable Class sourceEntityClass, @Nonnull Filter sourceEntityFilter, - @Nonnull List, RelationshipFilter, Class>> traversePaths, int offset, int count); + public abstract List findEntities(@Nullable Class sourceEntityClass, + @Nonnull Filter sourceEntityFilter, @Nonnull List traversePaths, int offset, int count); /** * Finds a list of relationships of a specific type based on the given relationship filter and source entity filter. diff --git a/dao-impl/neo4j-dao/src/main/java/com/linkedin/metadata/dao/Neo4jQueryDAO.java b/dao-impl/neo4j-dao/src/main/java/com/linkedin/metadata/dao/Neo4jQueryDAO.java index 3257f879e..9c40df873 100644 --- a/dao-impl/neo4j-dao/src/main/java/com/linkedin/metadata/dao/Neo4jQueryDAO.java +++ b/dao-impl/neo4j-dao/src/main/java/com/linkedin/metadata/dao/Neo4jQueryDAO.java @@ -15,7 +15,6 @@ import java.util.stream.StreamSupport; import javax.annotation.Nonnull; import javax.annotation.Nullable; -import org.javatuples.Triplet; import org.neo4j.driver.Driver; import org.neo4j.driver.Record; import org.neo4j.driver.Session; @@ -114,10 +113,8 @@ public List findEntities( - @Nullable Class sourceEntityClass, @Nonnull Filter sourceEntityFilter, - @Nonnull List, RelationshipFilter, Class>> traversePaths, int offset, - int count) { + public List findEntities(@Nullable Class sourceEntityClass, + @Nonnull Filter sourceEntityFilter, @Nonnull List traversePaths, int offset, int count) { if (sourceEntityClass != null) { EntityValidator.validateEntitySchema(sourceEntityClass); } @@ -128,16 +125,18 @@ public , RelationshipFilter, Class> path : traversePaths) { + // for each path, construct substatement via relationship type + relationship filer + inter entity + for (TraversalPath path : traversePaths) { pathCounter++; // Cannot use the same relationship variable 'r' or 'dest' for multiple patterns - final String edgeType = getTypeOrEmptyString(path.getValue0()); - final String edgeCriteria = path.getValue1() == null ? "" : criterionToString(path.getValue1().getCriteria()); + final String edgeType = getTypeOrEmptyString(path.getRelationship()); + final String edgeCriteria = + path.getRelationshipFilter() == null ? "" : criterionToString(path.getRelationshipFilter().getCriteria()); final RelationshipDirection relationshipDirection = - path.getValue1() == null ? RelationshipDirection.UNDIRECTED : path.getValue1().getDirection(); - final String destType = getTypeOrEmptyString(path.getValue2()); + path.getRelationshipFilter() == null ? RelationshipDirection.UNDIRECTED + : path.getRelationshipFilter().getDirection(); + final String destType = getTypeOrEmptyString(path.getIntermediateEntity()); String subTemplate = "-[r%d%s %s]-(dest%d%s)"; if (relationshipDirection == RelationshipDirection.INCOMING) { diff --git a/dao-impl/neo4j-dao/src/test/java/com/linkedin/metadata/dao/Neo4jQueryDAOTest.java b/dao-impl/neo4j-dao/src/test/java/com/linkedin/metadata/dao/Neo4jQueryDAOTest.java index cd2089944..655539185 100644 --- a/dao-impl/neo4j-dao/src/test/java/com/linkedin/metadata/dao/Neo4jQueryDAOTest.java +++ b/dao-impl/neo4j-dao/src/test/java/com/linkedin/metadata/dao/Neo4jQueryDAOTest.java @@ -22,7 +22,6 @@ import java.util.List; import java.util.Map; import java.util.stream.Collectors; -import org.javatuples.Triplet; import org.neo4j.driver.Driver; import org.neo4j.driver.GraphDatabase; import org.neo4j.driver.Record; @@ -340,25 +339,21 @@ public void testFindEntitiesViaTraversePathes() throws Exception { Filter sourceFilter = newFilter("urn", urn1.toString()); // use case: return all the datasets owned by corpgroup1 - List paths = new ArrayList(); - paths.add( - Triplet.with(RelationshipFoo.class, createRelationshipFilter(EMPTY_FILTER, RelationshipDirection.OUTGOING), - EntityBar.class)); - paths.add( - Triplet.with(RelationshipBar.class, createRelationshipFilter(EMPTY_FILTER, RelationshipDirection.INCOMING), - EntityBaz.class)); + List paths = new ArrayList<>(); + paths.add(new BaseQueryDAO.TraversalPath(RelationshipFoo.class, + createRelationshipFilter(EMPTY_FILTER, RelationshipDirection.OUTGOING), EntityBar.class)); + paths.add(new BaseQueryDAO.TraversalPath(RelationshipBar.class, + createRelationshipFilter(EMPTY_FILTER, RelationshipDirection.INCOMING), EntityBaz.class)); List result = _dao.findEntities(EntityFoo.class, sourceFilter, paths, 0, 10); assertEquals(result.size(), 1); assertEquals(result.get(0), entity3); // use case: return all the datasets owned by corpgroup1 - List paths2 = new ArrayList(); - paths2.add( - Triplet.with(RelationshipFoo.class, createRelationshipFilter(EMPTY_FILTER, RelationshipDirection.UNDIRECTED), - EntityBar.class)); - paths2.add( - Triplet.with(RelationshipBar.class, createRelationshipFilter(EMPTY_FILTER, RelationshipDirection.UNDIRECTED), - EntityBaz.class)); + List paths2 = new ArrayList<>(); + paths2.add(new BaseQueryDAO.TraversalPath(RelationshipFoo.class, + createRelationshipFilter(EMPTY_FILTER, RelationshipDirection.UNDIRECTED), EntityBar.class)); + paths2.add(new BaseQueryDAO.TraversalPath(RelationshipBar.class, + createRelationshipFilter(EMPTY_FILTER, RelationshipDirection.UNDIRECTED), EntityBaz.class)); List result2 = _dao.findEntities(EntityFoo.class, sourceFilter, paths2, 0, 10); assertEquals(result2, result); @@ -379,38 +374,37 @@ public void testFindEntitiesViaTraversePathes() throws Exception { RelationshipFoo relationshipFoo1To4 = new RelationshipFoo().setSource(urn1).setDestination(urn4); _writer.addRelationship(relationshipFoo1To4); - List paths3 = new ArrayList(); - paths3.add( - Triplet.with(RelationshipFoo.class, createRelationshipFilter(EMPTY_FILTER, RelationshipDirection.OUTGOING), - EntityBar.class)); - paths3.add( - Triplet.with(RelationshipBar.class, createRelationshipFilter(EMPTY_FILTER, RelationshipDirection.INCOMING), - EntityBaz.class)); + List paths3 = new ArrayList<>(); + paths3.add(new BaseQueryDAO.TraversalPath(RelationshipFoo.class, + createRelationshipFilter(EMPTY_FILTER, RelationshipDirection.OUTGOING), EntityBar.class)); + paths3.add(new BaseQueryDAO.TraversalPath(RelationshipBar.class, + createRelationshipFilter(EMPTY_FILTER, RelationshipDirection.INCOMING), EntityBaz.class)); List result3 = _dao.findEntities(EntityFoo.class, sourceFilter, paths3, 0, 10); assertEquals(result3.size(), 2); assertEquals(result3.get(0), entity5); assertEquals(result3.get(1), entity3); // test nulls - List paths4 = new ArrayList(); - paths4.add(Triplet.with(null, null, null)); + List paths4 = new ArrayList<>(); + paths4.add(new BaseQueryDAO.TraversalPath(null, null, null)); List result4 = _dao.findEntities(EntityFoo.class, sourceFilter, paths4, 0, 10); assertEquals(result4.size(), 2); assertEquals(result4.get(0), entity4); assertEquals(result4.get(1), entity2); // test partial nulls with entity - List paths5 = new ArrayList(); + List paths5 = new ArrayList<>(); paths5.add( - Triplet.with(null, createRelationshipFilter(EMPTY_FILTER, RelationshipDirection.OUTGOING), EntityBar.class)); + new BaseQueryDAO.TraversalPath(null, createRelationshipFilter(EMPTY_FILTER, RelationshipDirection.OUTGOING), + EntityBar.class)); List result5 = _dao.findEntities(EntityFoo.class, sourceFilter, paths5, 0, 10); assertEquals(result5.size(), 2); assertEquals(result5.get(0), entity4); assertEquals(result5.get(1), entity2); // test partial nulls with relationship - List paths6 = new ArrayList(); - paths6.add(Triplet.with(null, null, EntityBar.class)); + List paths6 = new ArrayList<>(); + paths6.add(new BaseQueryDAO.TraversalPath(null, null, EntityBar.class)); List result6 = _dao.findEntities(EntityFoo.class, sourceFilter, paths6, 0, 10); assertEquals(result6.size(), 2); assertEquals(result6.get(0), entity4); @@ -511,15 +505,17 @@ public void testFindNodesInPath() throws Exception { // Get reports roll-up - 2 levels Filter sourceFilter = newFilter("urn", urn1.toString()); RelationshipFilter relationshipFilter = createRelationshipFilter(EMPTY_FILTER, RelationshipDirection.INCOMING); - List> paths = _dao.findPaths(EntityFoo.class, sourceFilter, null, - EMPTY_FILTER, RelationshipFoo.class, relationshipFilter, 1, 2, -1, -1); + List> paths = + _dao.findPaths(EntityFoo.class, sourceFilter, null, EMPTY_FILTER, RelationshipFoo.class, relationshipFilter, 1, + 2, -1, -1); assertEquals(paths.size(), 5); assertEquals(paths.stream().filter(l -> l.size() == 3).collect(Collectors.toList()).size(), 2); assertEquals(paths.stream().filter(l -> l.size() == 5).collect(Collectors.toList()).size(), 3); // Get reports roll-up - 1 level - paths = _dao.findPaths(EntityFoo.class, sourceFilter, null, - EMPTY_FILTER, RelationshipFoo.class, relationshipFilter, 1, 1, -1, -1); + paths = + _dao.findPaths(EntityFoo.class, sourceFilter, null, EMPTY_FILTER, RelationshipFoo.class, relationshipFilter, 1, + 1, -1, -1); assertEquals(paths.size(), 2); assertEquals(paths.stream().filter(l -> l.size() == 3).collect(Collectors.toList()).size(), 2); assertEquals(paths.stream().filter(l -> l.size() == 5).collect(Collectors.toList()).size(), 0); @@ -601,9 +597,8 @@ public void testRunFreeFormQuery() throws Exception { String cypherQuery = "MATCH (n {value:\"foo\"}) RETURN n ORDER BY n.urn"; List result = _dao.runFreeFormQuery(cypherQuery); - List nodes = result.stream() - .map(record -> _dao.nodeRecordToEntity(EntityFoo.class, record)) - .collect(Collectors.toList()); + List nodes = + result.stream().map(record -> _dao.nodeRecordToEntity(EntityFoo.class, record)).collect(Collectors.toList()); assertEquals(nodes.size(), 2); assertEquals(nodes.get(0), entity1); assertEquals(nodes.get(1), entity2);