Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Problem with aliasing expression for "a depth-first" query using Esqueleto's "withRecursive" #388

Open
ciukstar opened this issue Feb 7, 2024 · 9 comments

Comments

@ciukstar
Copy link

ciukstar commented Feb 7, 2024

I'm trying to run the following query with a recursive common table expressions using Esqueleto's withRecursive but received an unexpected error at runtime.
Is there a way to fix this?

 do   
    tags <- runDB $ select $ do
        cte <- withRecursive
            (do
                  x <- from $ table @SignTag
                  where_ $ isNothing_ $ x ^. SignTagGroup
                  let level = val (0 :: Int)
                  return (level,x)
            )
            unionAll_
            (\parent -> do
                  (l,_) :& x <- from $ parent
                      `innerJoin` table @SignTag `on` (\((_, p) :& x) -> just (p ^. SignTagId) ==. x ^. SignTagGroup)
                  let level = l +. val 1
                  orderBy [desc level] -- ^ Here is the issue, I think
                  return (level,x)
            )
        from cte

Runtime error:

[Error#yesod-core] SQLite3 returned ErrorError while attempting to perform prepare "WITH RECURSIVE \"cte\" AS (SELECT ? AS \"v\", \"sign_tag\".\"id\" AS \"v2_id\", \"sign_tag\".\"name\" AS \"v2_name\", \"sign_tag\".\"descr\" AS \"v2_descr\", \"sign_tag\".\"group\" AS \"v2_group\"\nFROM \"sign_tag\"\nWHERE \"sign_tag\".\"group\" IS NULL\nUNION ALL\nSELECT \"cte\".\"v\", \"sign_tag2\".\"id\", \"sign_tag2\".\"name\", \"sign_tag2\".\"descr\", \"sign_tag2\".\"group\"\nFROM \"cte\" INNER JOIN \"sign_tag\" AS \"sign_tag2\" ON \"cte\".\"v2_id\" = \"sign_tag2\".\"group\"\nORDER BY \"cte\".\"v\" + ? DESC\n)\nSELECT \"cte\".\"v\", \"cte\".\"v2_id\", \"cte\".\"v2_name\", \"cte\".\"v2_descr\", \"cte\".\"v2_group\"\nFROM \"cte\"\n": 1st ORDER BY term does not match any column in the result set

The generated query is:

WITH RECURSIVE "cte" AS (
SELECT ? AS "v"
  , "sign_tag"."id" AS "v2_id"
  , "sign_tag"."name" AS "v2_name"
  , "sign_tag"."descr" AS "v2_descr"
  , "sign_tag"."group" AS "v2_group"
FROM "sign_tag"
WHERE "sign_tag"."group" IS NULL
UNION ALL
SELECT ("cte"."v" + ?)
     , "sign_tag2"."id"
     , "sign_tag2"."name"
     , "sign_tag2"."descr"
     , "sign_tag2"."group"
FROM "cte" INNER JOIN "sign_tag" AS "sign_tag2" ON "cte"."v2_id" = "sign_tag2"."group"
ORDER BY "cte"."v" + ? DESC
)
SELECT "cte"."v", "cte"."v2_id", "cte"."v2_name", "cte"."v2_descr", "cte"."v2_group"
FROM "cte"

with 0 and 1 as arguments.

@belevy
Copy link
Collaborator

belevy commented Feb 8, 2024

Is order by v + 1 legal? If you just order by v does it work?

@ciukstar
Copy link
Author

ciukstar commented Feb 8, 2024

Is order by v + 1 legal? If you just order by v does it work?

ORDER BY "cte"."v" + ? DESC this is what is generated.

I tried this too with the same RUNTIME error (compiles without problems):

...
            unionAll_
            (\parent -> do
                  (l,_) :& x <- from $ parent
                      `innerJoin` table @SignTag `on` (\((_, p) :& x) -> just (p ^. SignTagId) ==. x ^. SignTagGroup)
                  let level = l +. val 1
                  orderBy [desc l]
                  return (level,x)
            )
...

I'm trying to do something like what is described in 3.4. Controlling Depth-First Versus Breadth-First Search Of a Tree Using ORDER BY:

But if we change the ORDER BY clause to add the "DESC" modifier, that will cause lower levels in the organization (with larger "level" values) to be processed first by the recursive-select, resulting in a depth-first search:

WITH RECURSIVE
  under_alice(name,level) AS (
    VALUES('Alice',0)
    UNION ALL
    SELECT org.name, under_alice.level+1
      FROM org JOIN under_alice ON org.boss=under_alice.name
     ORDER BY 2 DESC
  )
SELECT substr('..........',1,level*3) || name FROM under_alice;
The output of this revised query is:

Alice
...Bob
......Dave
......Emma
...Cindy
......Fred
......Gail
When the ORDER BY clause is omitted from the recursive-select, the queue behaves as a FIFO, which results in a breadth-first search.

Maybe there is a way to order by column position? I didn't find anything about this.

@belevy
Copy link
Collaborator

belevy commented Feb 8, 2024

Right so it looks like in order to use an expression like v+1 in an order by you need to wrap that in parentheses. That seems to be not specific to how recursive CTEs work. That said sorting by a value v and a value v + 1 should behave identically and should function to work around the error.

@ciukstar
Copy link
Author

ciukstar commented Feb 9, 2024

to wrap that in parentheses

If I may, how to wrap the expression in parentheses?

This clearly doesn't work:)


                  let level = ( l +. val 1 )
                  orderBy [desc level]
                  return (level,x)

@belevy
Copy link
Collaborator

belevy commented Feb 9, 2024

Please read my entire reply.

The error you are experiencing is more general than just recursive CTEs and would require a more general fix.

However, you don't actually need to sort on your level variable since sorting on what you have named as l should function identically.

@ciukstar
Copy link
Author

ciukstar commented Feb 9, 2024

Please read my entire reply.

The error you are experiencing is more general than just recursive CTEs and would require a more general fix.

However, you don't actually need to sort on your level variable since sorting on what you have named as l should function identically.

It does not work for:

                  let level = l +. val 1
                  orderBy [desc l]
                  return (level,x)

Error:

SQLite3 returned ErrorError while attempting to perform prepare "WITH RECURSIVE \"cte\" AS (SELECT ? AS \"v\", \"sign_tag\".\"id\" AS \"v2_id\", \"sign_tag\".\"name\" AS \"v2_name\", \"sign_tag\".\"descr\" AS \"v2_descr\", \"sign_tag\".\"group\" AS \"v2_group\"\nFROM \"sign_tag\"\nWHERE \"sign_tag\".\"group\" IS NULL\nUNION ALL\nSELECT (\"cte\".\"v\" + ?), \"sign_tag2\".\"id\", \"sign_tag2\".\"name\", \"sign_tag2\".\"descr\", \"sign_tag2\".\"group\"\nFROM \"cte\" INNER JOIN \"sign_tag\" AS \"sign_tag2\" ON \"cte\".\"v2_id\" = \"sign_tag2\".\"group\"\nORDER BY \"cte\".\"v\" DESC\n)\nSELECT \"cte\".\"v\", \"cte\".\"v2_id\", \"cte\".\"v2_name\", \"cte\".\"v2_descr\", \"cte\".\"v2_group\"\nFROM \"cte\"\n": 1st ORDER BY term does not match any column in the result set

Generated SQL:

WITH RECURSIVE "cte" AS (
SELECT ? AS "v"
     , "sign_tag"."id" AS "v2_id"
     , "sign_tag"."name" AS "v2_name"
     , "sign_tag"."descr" AS "v2_descr"
     , "sign_tag"."group" AS "v2_group"
FROM "sign_tag"
WHERE "sign_tag"."group" IS NULL
UNION ALL
SELECT ("cte"."v" + ?)
       , "sign_tag2"."id"
       , "sign_tag2"."name"
       , "sign_tag2"."descr"
       , "sign_tag2"."group"
FROM "cte" INNER JOIN "sign_tag" AS "sign_tag2" ON "cte"."v2_id" = "sign_tag2"."group"
ORDER BY "cte"."v" DESC
)
SELECT "cte"."v", "cte"."v2_id", "cte"."v2_name", "cte"."v2_descr", "cte"."v2_group"
FROM "cte"

@belevy
Copy link
Collaborator

belevy commented Feb 9, 2024

You could probably just orderBy [desc (val 1)] but that seems to be sqlite specific syntax.

Ultimately the bug that needs fixing is to properly parenthesize expressions in order by clauses.

As an aside, the whole order of the rows popping off the work queue thing is super low level and not all that intuitive to me(but I don't really use recursive CTEs all that much). I would personally prefer to keep IDs to reference who my parent is rather than relying on the order of the results to build a tree like that.

@ciukstar
Copy link
Author

ciukstar commented Feb 9, 2024

You could probably just orderBy [desc (val 1)] but that seems to be sqlite specific syntax.

Unfortunately this doesn't work either:


                  let level = l +. val 1
                  orderBy [desc (val (1 :: Int))]
                  return (level,x)

Error:

[Error#yesod-core] SQLite3 returned ErrorError while attempting to perform prepare "WITH RECURSIVE \"cte\" AS (SELECT ? AS \"v\", \"sign_tag\".\"id\" AS \"v2_id\", \"sign_tag\".\"name\" AS \"v2_name\", \"sign_tag\".\"descr\" AS \"v2_descr\", \"sign_tag\".\"group\" AS \"v2_group\"\nFROM \"sign_tag\"\nWHERE \"sign_tag\".\"group\" IS NULL\nUNION ALL\nSELECT (\"cte\".\"v\" + ?), \"sign_tag2\".\"id\", \"sign_tag2\".\"name\", \"sign_tag2\".\"descr\", \"sign_tag2\".\"group\"\nFROM \"cte\" INNER JOIN \"sign_tag\" AS \"sign_tag2\" ON \"cte\".\"v2_id\" = \"sign_tag2\".\"group\"\nORDER BY ? DESC\n)\nSELECT \"cte\".\"v\", \"cte\".\"v2_id\", \"cte\".\"v2_name\", \"cte\".\"v2_descr\", \"cte\".\"v2_group\"\nFROM \"cte\"\n": 1st ORDER BY term does not match any column in the result set @(yesod-core-1.6.25.1

Generated query:

[Debug#SQL] 

WITH RECURSIVE "cte" AS (SELECT ? AS "v", "sign_tag"."id" AS "v2_id", "sign_tag"."name" AS "v2_name", "sign_tag"."descr" AS "v2_descr", "sign_tag"."group" AS "v2_group"
FROM "sign_tag"
WHERE "sign_tag"."group" IS NULL
UNION ALL
SELECT ("cte"."v" + ?), "sign_tag2"."id", "sign_tag2"."name", "sign_tag2"."descr", "sign_tag2"."group"
FROM "cte" INNER JOIN "sign_tag" AS "sign_tag2" ON "cte"."v2_id" = "sign_tag2"."group"
ORDER BY ? DESC
)
SELECT "cte"."v", "cte"."v2_id", "cte"."v2_name", "cte"."v2_descr", "cte"."v2_group"
FROM "cte"
;

 [PersistInt64 0,PersistInt64 1,PersistInt64 1]

@parsonsmatt
Copy link
Collaborator

I think the work-around for this is to use unsafeSqlValue - probably something like:

sqliteOrderByColumnIndex 
    :: Int 
    -> (forall a. SqlExpr (Value a) -> SqlExpr OrderBy) 
    -> SqlExpr OrderBy
sqliteOrderByColumnIndex i ascOrDesc =
    ascOrDesc 
        ( unsafeSqlValue 
            $ Data.Text.Lazy.Builder.Int.decimal i
            :: SqlExpr (Value Int)
        )

Then you'd write orderBy [sqliteOrderByColumnIndex 1 asc].

I tried to write a minimal test reproduction, assuming the issue was orderBy [asc (x +. val 1)], but this test passes:

    itDb "works with an expression" $ do
        let p1 = Point 1 2 "Hello"
            p2 = Point 2 1 "Goodbye"
        p1k <- insert p1
        p2k <- insert p2
        entityPoints <-
            select $ do
                p <- Experimental.from table
                orderBy [asc (p ^. PointX +. val 1)]
                pure p

        asserting $ do
            fmap entityKey entityPoints
                `shouldBe`
                    [p1k, p2k]

If I implement a test for the behavior above, I get failures on all three backends:

Failures:

  test/Common/Test/CTE.hs:39:13: 
  1) Esqueleto.SQLite.Esqueleto.CTE.withRecursive, depth first search, works
       uncaught exception: SqliteException
       SQLite3 returned ErrorError while attempting to perform prepare "WITH RECURSIVE \"cte\" AS (SELECT ? AS \"v\", \"SignTag\".\"id\" AS \"v2_id\", \"SignTag\".\"group\" 
AS \"v2_group\"\nFROM \"SignTag\"\nWHERE \"SignTag\".\"group\" IS NULL\nUNION ALL\nSELECT (\"cte\".\"v\" + ?), \"SignTag2\".\"id\", \"SignTag2\".\"group\"\nFROM \"cte\" INNE
R JOIN \"SignTag\" AS \"SignTag2\" ON \"cte\".\"v2_id\" = \"SignTag2\".\"group\"\nORDER BY \"cte\".\"v\" + ? DESC\n)\nSELECT \"cte\".\"v\", \"cte\".\"v2_id\", \"cte\".\"v2_g
roup\"\nFROM \"cte\"\n": 1st ORDER BY term does not match any column in the result set

  To rerun use: --match "/Esqueleto/SQLite/Esqueleto/CTE/withRecursive/depth first search/works/"

  test/Common/Test/CTE.hs:39:13: 
  2) Esqueleto.MySQL.Esqueleto.CTE.withRecursive, depth first search, works
       uncaught exception: MySQLError
       ConnectionError {errFunction = "query", errNumber = 1250, errMessage = "Table 'cte' from one of the SELECTs cannot be used in global ORDER clause"}

  To rerun use: --match "/Esqueleto/MySQL/Esqueleto/CTE/withRecursive/depth first search/works/"

  test/Common/Test/CTE.hs:39:13: 
  3) Esqueleto.Postgresql.Esqueleto.CTE.withRecursive, depth first search, works
       uncaught exception: SqlError
       SqlError {sqlState = "0A000", sqlExecStatus = FatalError, sqlErrorMsg = "ORDER BY in a recursive query is not implemented", sqlErrorDetail = "", sqlErrorHint = ""}

  To rerun use: --match "/Esqueleto/Postgresql/Esqueleto/CTE/withRecursive/depth first search/works/"

Randomized with seed 1042962468

Finished in 0.8472 seconds
629 examples, 3 failures, 9 pending

If I look at the generated query more carefully, I see this:

WITH RECURSIVE cte AS (
    SELECT ? as v, SignTag.id AS v2_id, SignTag.group AS v2_group
    FROM SignTag 
    WHERE SignTag.group IS NOTHING
  UNION ALL 
    SELECT (cte.v + ?), SignTag2.id, SignTag2.group
    FROM cte INNER JOIN SignTag as SignTag2 ON cte.v2_id = SignTag2.group
    ORDER BY cte.v + ? DESC)
SELECT cte.v, cte.v2_id, cte.v2_group

The second SELECT (cte.v + ?) and the ORDER BY (cte.v + ? DESC) are apparently distinct according to SQLite - at query preparation time, it can't ensure that the two ? parameters end up being the same thing.

If we use a literal 1 instead of a val 1, then it does work:

    describe "withRecursive" $ do
        describe "depth first search" $ do
            itDb "works" $ do
                tags <- select $ do
                    cte <- withRecursive
                        (do
                              x <- from $ table @SignTag
                              where_ $ isNothing_ $ x ^. SignTagGroup
                              let level = val (0 :: Int)
                              return (level,x)
                        )
                        unionAll_
                        (\parent -> do
                              (l,_) :& x <- from $ parent
                                  `innerJoin` table @SignTag `on` (\((_, p) :& x) -> just (p ^. SignTagId) ==. x ^. SignTagGroup)
                              let level = l +. literalInt 1
                              orderBy [desc level] -- ^ Here is the issue, I think
                              return (level,x)
                        )
                    from cte
                asserting $ do
                    tags `shouldBe` []

literalInt :: Int -> SqlExpr (Value Int)
literalInt = unsafeSqlValue . TLBI.decimal

This passes on SQLite tests, but still fails on MySQL and Postgres.

Likewise, using literalInt in the orderBy works:

    describe "withRecursive" $ do
        describe "depth first search" $ do
            itDb "works" $ do
                tags <- select $ do
                    cte <- withRecursive
                        (do
                              x <- from $ table @SignTag
                              where_ $ isNothing_ $ x ^. SignTagGroup
                              let level = val (0 :: Int)
                              return (level,x)
                        )
                        unionAll_
                        (\parent -> do
                              (l,_) :& x <- from $ parent
                                  `innerJoin` table @SignTag `on` (\((_, p) :& x) -> just (p ^. SignTagId) ==. x ^. SignTagGroup)
                              let level = l +. val 1
                              orderBy [desc (literalInt 1)] -- ^ Here is the issue, I think
                              return (level,x)
                        )
                    from cte
                asserting $ do
                    tags `shouldBe` []

literalInt :: Int -> SqlExpr (Value Int)
literalInt = unsafeSqlValue . TLBI.decimal

I think the fix is to add a function to the SQLite module which documents this and provides a somewhat-safe interface to this behavior.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

3 participants