2013-12-17 10:40:31 +01:00
|
|
|
|
|
|
|
Here are some tests taken from the SQL in the postgres manual. Almost
|
|
|
|
all of the postgres specific syntax has been skipped, this can be
|
|
|
|
revisited when the dialect support is added.
|
|
|
|
|
|
|
|
> module Language.SQL.SimpleSQL.Postgres (postgresTests) where
|
|
|
|
|
|
|
|
> import Language.SQL.SimpleSQL.TestTypes
|
2013-12-17 11:16:03 +01:00
|
|
|
> --import Language.SQL.SimpleSQL.Syntax
|
2013-12-17 10:40:31 +01:00
|
|
|
|
|
|
|
> postgresTests :: TestItem
|
2013-12-17 11:16:03 +01:00
|
|
|
> postgresTests = Group "postgresTests" $ map ParseQueryExpr $
|
|
|
|
|
|
|
|
lexical syntax section
|
|
|
|
|
|
|
|
TODO: get all the commented out tests working
|
|
|
|
|
|
|
|
> [-- "SELECT 'foo'\n\
|
|
|
|
> -- \'bar';" -- this should parse as select 'foobar'
|
|
|
|
> -- ,
|
|
|
|
> "SELECT name, (SELECT max(pop) FROM cities WHERE cities.state = states.name)\n\
|
|
|
|
> \ FROM states;"
|
|
|
|
> ,"SELECT ROW(1,2.5,'this is a test');"
|
2013-12-17 10:40:31 +01:00
|
|
|
|
2013-12-17 11:24:37 +01:00
|
|
|
> ,"SELECT ROW(t.*, 42) FROM t;"
|
2013-12-17 11:16:03 +01:00
|
|
|
> ,"SELECT ROW(t.f1, t.f2, 42) FROM t;"
|
|
|
|
> ,"SELECT getf1(CAST(ROW(11,'this is a test',2.5) AS myrowtype));"
|
2013-12-17 10:40:31 +01:00
|
|
|
|
2013-12-17 11:16:03 +01:00
|
|
|
> ,"SELECT ROW(1,2.5,'this is a test') = ROW(1, 3, 'not the same');"
|
2013-12-17 10:40:31 +01:00
|
|
|
|
2013-12-17 11:16:03 +01:00
|
|
|
> ,"SELECT ROW(table.*) IS NULL FROM table;"
|
2013-12-17 10:40:31 +01:00
|
|
|
|
2013-12-17 11:16:03 +01:00
|
|
|
> ,"SELECT true OR somefunc();"
|
2013-12-17 10:40:31 +01:00
|
|
|
|
2013-12-17 11:16:03 +01:00
|
|
|
> ,"SELECT somefunc() OR true;"
|
2013-12-17 10:40:31 +01:00
|
|
|
|
2013-12-17 11:16:03 +01:00
|
|
|
queries section
|
2013-12-17 10:40:31 +01:00
|
|
|
|
2013-12-17 11:16:03 +01:00
|
|
|
> ,"SELECT * FROM t1 CROSS JOIN t2;"
|
|
|
|
> ,"SELECT * FROM t1 INNER JOIN t2 ON t1.num = t2.num;"
|
|
|
|
> ,"SELECT * FROM t1 INNER JOIN t2 USING (num);"
|
|
|
|
> ,"SELECT * FROM t1 NATURAL INNER JOIN t2;"
|
|
|
|
> ,"SELECT * FROM t1 LEFT JOIN t2 ON t1.num = t2.num;"
|
|
|
|
> ,"SELECT * FROM t1 LEFT JOIN t2 USING (num);"
|
|
|
|
> ,"SELECT * FROM t1 RIGHT JOIN t2 ON t1.num = t2.num;"
|
|
|
|
> ,"SELECT * FROM t1 FULL JOIN t2 ON t1.num = t2.num;"
|
|
|
|
> ,"SELECT * FROM t1 LEFT JOIN t2 ON t1.num = t2.num AND t2.value = 'xxx';"
|
|
|
|
> ,"SELECT * FROM t1 LEFT JOIN t2 ON t1.num = t2.num WHERE t2.value = 'xxx';"
|
|
|
|
|
2013-12-17 11:27:00 +01:00
|
|
|
> ,"SELECT * FROM some_very_long_table_name s JOIN another_fairly_long_name a ON s.id = a.num;"
|
|
|
|
> ,"SELECT * FROM people AS mother JOIN people AS child ON mother.id = child.mother_id;"
|
2013-12-17 11:16:03 +01:00
|
|
|
> ,"SELECT * FROM my_table AS a CROSS JOIN my_table AS b;"
|
|
|
|
> ,"SELECT * FROM (my_table AS a CROSS JOIN my_table) AS b;"
|
2013-12-17 11:45:32 +01:00
|
|
|
> ,"SELECT * FROM getfoo(1) AS t1;"
|
2013-12-17 11:33:33 +01:00
|
|
|
> ,"SELECT * FROM foo\n\
|
2013-12-17 11:16:03 +01:00
|
|
|
> \ WHERE foosubid IN (\n\
|
|
|
|
> \ SELECT foosubid\n\
|
|
|
|
> \ FROM getfoo(foo.fooid) z\n\
|
|
|
|
> \ WHERE z.fooid = foo.fooid\n\
|
2013-12-17 11:33:33 +01:00
|
|
|
> \ );"
|
2013-12-17 11:16:03 +01:00
|
|
|
> {-,"SELECT *\n\
|
|
|
|
> \ FROM dblink('dbname=mydb', 'SELECT proname, prosrc FROM pg_proc')\n\
|
|
|
|
> \ AS t1(proname name, prosrc text)\n\
|
2013-12-17 11:33:33 +01:00
|
|
|
> \ WHERE proname LIKE 'bytea%';"-} -- types in the alias??
|
2013-12-17 11:16:03 +01:00
|
|
|
|
2013-12-17 11:45:32 +01:00
|
|
|
> ,"SELECT * FROM foo, LATERAL (SELECT * FROM bar WHERE bar.id = foo.bar_id) ss;" -- lateral
|
2013-12-17 11:16:03 +01:00
|
|
|
> ,"SELECT * FROM foo, bar WHERE bar.id = foo.bar_id;"
|
|
|
|
|
|
|
|
> {-,"SELECT p1.id, p2.id, v1, v2\n\
|
|
|
|
> \FROM polygons p1, polygons p2,\n\
|
|
|
|
> \ LATERAL vertices(p1.poly) v1,\n\
|
|
|
|
> \ LATERAL vertices(p2.poly) v2\n\
|
2013-12-17 11:45:32 +01:00
|
|
|
> \WHERE (v1 <-> v2) < 10 AND p1.id != p2.id;"-} -- <-> operator?
|
2013-12-17 11:16:03 +01:00
|
|
|
|
|
|
|
> {-,"SELECT p1.id, p2.id, v1, v2\n\
|
|
|
|
> \FROM polygons p1 CROSS JOIN LATERAL vertices(p1.poly) v1,\n\
|
|
|
|
> \ polygons p2 CROSS JOIN LATERAL vertices(p2.poly) v2\n\
|
|
|
|
> \WHERE (v1 <-> v2) < 10 AND p1.id != p2.id;"-}
|
|
|
|
|
2013-12-17 11:45:32 +01:00
|
|
|
> ,"SELECT m.name\n\
|
2013-12-17 11:16:03 +01:00
|
|
|
> \FROM manufacturers m LEFT JOIN LATERAL get_product_names(m.id) pname ON true\n\
|
2013-12-17 11:45:32 +01:00
|
|
|
> \WHERE pname IS NULL;"
|
2013-12-17 11:16:03 +01:00
|
|
|
|
|
|
|
|
|
|
|
> ,"SELECT * FROM fdt WHERE c1 > 5"
|
|
|
|
|
|
|
|
> ,"SELECT * FROM fdt WHERE c1 IN (1, 2, 3)"
|
|
|
|
|
|
|
|
> ,"SELECT * FROM fdt WHERE c1 IN (SELECT c1 FROM t2)"
|
|
|
|
|
|
|
|
> ,"SELECT * FROM fdt WHERE c1 IN (SELECT c3 FROM t2 WHERE c2 = fdt.c1 + 10)"
|
|
|
|
|
|
|
|
> ,"SELECT * FROM fdt WHERE c1 BETWEEN (SELECT c3 FROM t2 WHERE c2 = fdt.c1 + 10) AND 100"
|
|
|
|
|
|
|
|
> ,"SELECT * FROM fdt WHERE EXISTS (SELECT c1 FROM t2 WHERE c2 > fdt.c1)"
|
|
|
|
|
|
|
|
> ,"SELECT * FROM test1;"
|
|
|
|
|
|
|
|
> ,"SELECT x FROM test1 GROUP BY x;"
|
|
|
|
> ,"SELECT x, sum(y) FROM test1 GROUP BY x;"
|
|
|
|
> ,"SELECT product_id, p.name, (sum(s.units) * p.price) AS sales\n\
|
|
|
|
> \ FROM products p LEFT JOIN sales s USING (product_id)\n\
|
|
|
|
> \ GROUP BY product_id, p.name, p.price;"
|
|
|
|
|
|
|
|
> ,"SELECT x, sum(y) FROM test1 GROUP BY x HAVING sum(y) > 3;"
|
|
|
|
> ,"SELECT x, sum(y) FROM test1 GROUP BY x HAVING x < 'c';"
|
|
|
|
> {-,"SELECT product_id, p.name, (sum(s.units) * (p.price - p.cost)) AS profit\n\
|
|
|
|
> \ FROM products p LEFT JOIN sales s USING (product_id)\n\
|
|
|
|
> \ WHERE s.date > CURRENT_DATE - INTERVAL '4 weeks'\n\
|
|
|
|
> \ GROUP BY product_id, p.name, p.price, p.cost\n\
|
|
|
|
> \ HAVING sum(p.price * s.units) > 5000;"-} -- interval syntax?
|
|
|
|
|
|
|
|
> ,"SELECT a, b, c FROM t"
|
|
|
|
|
|
|
|
> ,"SELECT tbl1.a, tbl2.a, tbl1.b FROM t"
|
|
|
|
|
|
|
|
> ,"SELECT tbl1.*, tbl2.a FROM t"
|
|
|
|
|
|
|
|
> ,"SELECT a AS value, b + c AS sum FROM t"
|
|
|
|
|
|
|
|
> --,"SELECT a \"value\", b + c AS sum FROM t" -- quoted identifier
|
|
|
|
|
|
|
|
> ,"SELECT DISTINCT select_list t"
|
|
|
|
|
|
|
|
> --,"VALUES (1, 'one'), (2, 'two'), (3, 'three');" -- values list
|
|
|
|
|
|
|
|
> ,"SELECT 1 AS column1, 'one' AS column2\n\
|
|
|
|
> \UNION ALL\n\
|
|
|
|
> \SELECT 2, 'two'\n\
|
|
|
|
> \UNION ALL\n\
|
|
|
|
> \SELECT 3, 'three';"
|
|
|
|
|
|
|
|
> --,"SELECT * FROM (VALUES (1, 'one'), (2, 'two'), (3, 'three')) AS t (num,letter);"
|
|
|
|
> -- values list
|
|
|
|
|
|
|
|
> ,"WITH regional_sales AS (\n\
|
|
|
|
> \ SELECT region, SUM(amount) AS total_sales\n\
|
|
|
|
> \ FROM orders\n\
|
|
|
|
> \ GROUP BY region\n\
|
|
|
|
> \ ), top_regions AS (\n\
|
|
|
|
> \ SELECT region\n\
|
|
|
|
> \ FROM regional_sales\n\
|
|
|
|
> \ WHERE total_sales > (SELECT SUM(total_sales)/10 FROM regional_sales)\n\
|
|
|
|
> \ )\n\
|
|
|
|
> \SELECT region,\n\
|
|
|
|
> \ product,\n\
|
|
|
|
> \ SUM(quantity) AS product_units,\n\
|
|
|
|
> \ SUM(amount) AS product_sales\n\
|
|
|
|
> \FROM orders\n\
|
|
|
|
> \WHERE region IN (SELECT region FROM top_regions)\n\
|
|
|
|
> \GROUP BY region, product;"
|
|
|
|
|
|
|
|
> {-,"WITH RECURSIVE t(n) AS (\n\
|
|
|
|
> \ VALUES (1)\n\
|
|
|
|
> \ UNION ALL\n\
|
|
|
|
> \ SELECT n+1 FROM t WHERE n < 100\n\
|
|
|
|
> \)\n\
|
|
|
|
> \SELECT sum(n) FROM t"-} -- full alias in cte
|
|
|
|
|
|
|
|
> {-,"WITH RECURSIVE included_parts(sub_part, part, quantity) AS (\n\
|
|
|
|
> \ SELECT sub_part, part, quantity FROM parts WHERE part = 'our_product'\n\
|
|
|
|
> \ UNION ALL\n\
|
|
|
|
> \ SELECT p.sub_part, p.part, p.quantity\n\
|
|
|
|
> \ FROM included_parts pr, parts p\n\
|
|
|
|
> \ WHERE p.part = pr.sub_part\n\
|
|
|
|
> \ )\n\
|
|
|
|
> \SELECT sub_part, SUM(quantity) as total_quantity\n\
|
|
|
|
> \FROM included_parts\n\
|
|
|
|
> \GROUP BY sub_part"
|
|
|
|
|
|
|
|
> ,"WITH RECURSIVE search_graph(id, link, data, depth) AS (\n\
|
|
|
|
> \ SELECT g.id, g.link, g.data, 1\n\
|
|
|
|
> \ FROM graph g\n\
|
|
|
|
> \ UNION ALL\n\
|
|
|
|
> \ SELECT g.id, g.link, g.data, sg.depth + 1\n\
|
|
|
|
> \ FROM graph g, search_graph sg\n\
|
|
|
|
> \ WHERE g.id = sg.link\n\
|
|
|
|
> \)\n\
|
|
|
|
> \SELECT * FROM search_graph;"
|
|
|
|
|
|
|
|
> ,"WITH RECURSIVE search_graph(id, link, data, depth, path, cycle) AS (\n\
|
|
|
|
> \ SELECT g.id, g.link, g.data, 1,\n\
|
|
|
|
> \ ARRAY[g.id],\n\
|
|
|
|
> \ false\n\
|
|
|
|
> \ FROM graph g\n\
|
|
|
|
> \ UNION ALL\n\
|
|
|
|
> \ SELECT g.id, g.link, g.data, sg.depth + 1,\n\
|
|
|
|
> \ path || g.id,\n\
|
|
|
|
> \ g.id = ANY(path)\n\
|
|
|
|
> \ FROM graph g, search_graph sg\n\
|
|
|
|
> \ WHERE g.id = sg.link AND NOT cycle\n\
|
|
|
|
> \)\n\
|
|
|
|
> \SELECT * FROM search_graph;"
|
|
|
|
|
|
|
|
> ,"WITH RECURSIVE search_graph(id, link, data, depth, path, cycle) AS (\n\
|
|
|
|
> \ SELECT g.id, g.link, g.data, 1,\n\
|
|
|
|
> \ ARRAY[ROW(g.f1, g.f2)],\n\
|
|
|
|
> \ false\n\
|
|
|
|
> \ FROM graph g\n\
|
|
|
|
> \ UNION ALL\n\
|
|
|
|
> \ SELECT g.id, g.link, g.data, sg.depth + 1,\n\
|
|
|
|
> \ path || ROW(g.f1, g.f2),\n\
|
|
|
|
> \ ROW(g.f1, g.f2) = ANY(path)\n\
|
|
|
|
> \ FROM graph g, search_graph sg\n\
|
|
|
|
> \ WHERE g.id = sg.link AND NOT cycle\n\
|
|
|
|
> \)\n\
|
|
|
|
> \SELECT * FROM search_graph;"
|
|
|
|
|
|
|
|
> ,"WITH RECURSIVE t(n) AS (\n\
|
|
|
|
> \ SELECT 1\n\
|
|
|
|
> \ UNION ALL\n\
|
|
|
|
> \ SELECT n+1 FROM t\n\
|
|
|
|
> \)\n\
|
|
|
|
> \SELECT n FROM t LIMIT 100;"-}
|
|
|
|
|
|
|
|
select page reference
|
2013-12-17 10:40:31 +01:00
|
|
|
|
2013-12-17 11:16:03 +01:00
|
|
|
> ,"SELECT f.title, f.did, d.name, f.date_prod, f.kind\n\
|
|
|
|
> \ FROM distributors d, films f\n\
|
|
|
|
> \ WHERE f.did = d.did"
|
|
|
|
|
|
|
|
> {-,"SELECT kind, sum(len) AS total\n\
|
|
|
|
> \ FROM films\n\
|
|
|
|
> \ GROUP BY kind\n\
|
|
|
|
> \ HAVING sum(len) < interval '5 hours';"-} -- interval syntax?
|
|
|
|
|
|
|
|
> ,"SELECT * FROM distributors ORDER BY name;"
|
|
|
|
> ,"SELECT * FROM distributors ORDER BY 2;"
|
|
|
|
|
|
|
|
> ,"SELECT distributors.name\n\
|
|
|
|
> \ FROM distributors\n\
|
|
|
|
> \ WHERE distributors.name LIKE 'W%'\n\
|
|
|
|
> \UNION\n\
|
|
|
|
> \SELECT actors.name\n\
|
|
|
|
> \ FROM actors\n\
|
|
|
|
> \ WHERE actors.name LIKE 'W%';"
|
|
|
|
|
2013-12-17 11:33:33 +01:00
|
|
|
> ,"WITH t AS (\n\
|
2013-12-17 11:16:03 +01:00
|
|
|
> \ SELECT random() as x FROM generate_series(1, 3)\n\
|
|
|
|
> \ )\n\
|
|
|
|
> \SELECT * FROM t\n\
|
|
|
|
> \UNION ALL\n\
|
2013-12-17 11:33:33 +01:00
|
|
|
> \SELECT * FROM t"
|
2013-12-17 11:16:03 +01:00
|
|
|
|
|
|
|
> {-,"WITH RECURSIVE employee_recursive(distance, employee_name, manager_name) AS (\n\
|
|
|
|
> \ SELECT 1, employee_name, manager_name\n\
|
|
|
|
> \ FROM employee\n\
|
|
|
|
> \ WHERE manager_name = 'Mary'\n\
|
|
|
|
> \ UNION ALL\n\
|
|
|
|
> \ SELECT er.distance + 1, e.employee_name, e.manager_name\n\
|
|
|
|
> \ FROM employee_recursive er, employee e\n\
|
|
|
|
> \ WHERE er.employee_name = e.manager_name\n\
|
|
|
|
> \ )\n\
|
|
|
|
> \SELECT distance, employee_name FROM employee_recursive;"-} -- with recursive, full cte alias
|
|
|
|
|
2013-12-17 11:45:32 +01:00
|
|
|
> ,"SELECT m.name AS mname, pname\n\
|
2013-12-17 11:16:03 +01:00
|
|
|
> \FROM manufacturers m, LATERAL get_product_names(m.id) pname;"
|
|
|
|
|
|
|
|
> ,"SELECT m.name AS mname, pname\n\
|
2013-12-17 11:45:32 +01:00
|
|
|
> \FROM manufacturers m LEFT JOIN LATERAL get_product_names(m.id) pname ON true;"
|
2013-12-17 11:16:03 +01:00
|
|
|
|
|
|
|
> ,"SELECT 2+2;"
|
|
|
|
|
|
|
|
> ,"SELECT distributors.* WHERE distributors.name = 'Westward';"
|
2013-12-17 10:40:31 +01:00
|
|
|
|
2013-12-17 11:16:03 +01:00
|
|
|
> ]
|
2013-12-17 10:40:31 +01:00
|
|
|
|
|
|
|
|
|
|
|
> {-f = mapM_ (putStrLn . either peFormattedError show . parseQueryExpr "" Nothing)
|
|
|
|
> ["SELECT * FROM t1 CROSS JOIN t2;"
|
|
|
|
> ,"SELECT * FROM t1 INNER JOIN t2 ON t1.num = t2.num;"
|
|
|
|
> ,"SELECT * FROM t1 INNER JOIN t2 USING (num);"
|
|
|
|
> ,"SELECT * FROM t1 NATURAL INNER JOIN t2;"
|
|
|
|
> ,"SELECT * FROM t1 LEFT JOIN t2 ON t1.num = t2.num;"
|
|
|
|
> ,"SELECT * FROM t1 LEFT JOIN t2 USING (num);"
|
|
|
|
> ,"SELECT * FROM t1 RIGHT JOIN t2 ON t1.num = t2.num;"
|
|
|
|
> ,"SELECT * FROM t1 FULL JOIN t2 ON t1.num = t2.num;"
|
|
|
|
> ,"SELECT * FROM t1 LEFT JOIN t2 ON t1.num = t2.num AND t2.value = 'xxx';"
|
|
|
|
> - ,"SELECT * FROM t1 LEFT JOIN t2 ON t1.num = t2.num WHERE t2.value = 'xxx';"]-}
|
|
|
|
|