Search code examples
node.jsneo4jneo4j-driver

Neo4J multiple unwinds unexpected behavior


I am trying to execute the following Neo4J query.

CREATE (c:Content)
WITH c
UNWIND $resources as resource
    MATCH (p:Person {userId: resource["userId"]})
    CREATE (r:Resource)
    CREATE (p)-[:CREATES]->(r)
    CREATE (c)-[:CONTAINS]->(r)
WITH c
UNWIND $tags as tag
    MATCH (t:Tag {name: tag})
    CREATE (c)-[:IS]->(t)
WITH c
UNWIND $goalCategories as category
    MATCH (atc:ATCategory {name: category})
    CREATE (c)-[:IS]->(atc)

$resources is an object list and $tags and $categories are list of strings. I have left the data manipulation part for simplicity.

When this query executes, it creates multiple relationships between c and atc which means that it repeats the category for each item in the tags. To my knowledge, since c is singular, the WITH c block should close the previous unwind but this does not happen. What's weird is that, this does not happen for the resources unwind.

Shortly, the code threats like the UNWIND $goalCategories as category is inside the UNWIND $tags as tag part. If the same happened between UNWIND $resources as resource and UNWIND $tags as tag, it would make sense, but I couldn't figure out why this is only happening for the latter part and not the prior part.

I would really appreciate the help.

Btw: I am executing the code with neo4j-driver on nodejs.


Solution

  • Running multiple consecutive UNWIND statements will lead to cartesian products. Essentially, all the nested UNWIND will be executed x times, where the x is the number of rows from the previous unwind. The WITH c does not close the previous UNWIND statement, you need to add any aggregation function. For example, you could use the count() aggregation function to effectively reduce the count of rows in between consecutive UNWIND statements to 1 (or close a previous UNWIND as you state).

    CREATE (c:Content)
    WITH c
    UNWIND $resources as resource
        MATCH (p:Person {userId: resource["userId"]})
        CREATE (r:Resource)
        CREATE (p)-[:CREATES]->(r)
        CREATE (c)-[:CONTAINS]->(r)
    WITH c, count(*) as dummy
    UNWIND $tags as tag
        MATCH (t:Tag {name: tag})
        CREATE (c)-[:IS]->(t)
    WITH c, count(*) AS dummy
    UNWIND $goalCategories as category
        MATCH (atc:ATCategory {name: category})
        CREATE (c)-[:IS]->(atc)