Compare commits
289 commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
f5dbc274be | ||
|
|
df753e8635 | ||
|
|
b4b2ca7d79 | ||
|
|
78f7d5282f | ||
|
|
024d17b11e | ||
|
|
7611871935 | ||
|
|
1b042687f4 | ||
|
|
a981ed9171 | ||
|
|
74cf16c134 | ||
|
|
2c8fc30b1d | ||
|
|
3906aa53c0 | ||
|
|
92e4e16b45 | ||
|
|
30b5fabe58 | ||
|
|
d74046c658 | ||
|
|
44494e61c0 | ||
|
|
d70e89ae3b | ||
|
|
67ea477a5c | ||
|
|
89f39be55c | ||
|
|
0d1fd0e901 | ||
|
|
675c94b294 | ||
|
|
03d96f5747 | ||
|
|
f7dbfba57c | ||
|
|
f0eb68f151 | ||
|
|
4d1f5f83b7 | ||
|
|
c2990597f1 | ||
|
|
695351e33c | ||
|
|
0cd81b5d9b | ||
|
|
c295db44c0 | ||
|
|
44ca426b78 | ||
|
|
e3fcb3e278 | ||
|
|
7e1fe8f558 | ||
|
|
13eb8fe859 | ||
|
|
4bc1d16f24 | ||
|
|
206f980093 | ||
|
|
3beaa6b2bf | ||
|
|
30a04975f5 | ||
|
|
4f5b0ed256 | ||
|
|
1681764830 | ||
|
|
6c0c66e371 | ||
|
|
2c793ce441 | ||
|
|
f05c7051e2 | ||
|
|
d086631e54 | ||
|
|
3ecac63bea | ||
|
|
8e0d6984bd | ||
|
|
94fae3437f | ||
|
|
316f36751f | ||
|
|
8ae93d91f6 | ||
|
|
5fa85400f0 | ||
|
|
045634fd3c | ||
|
|
81167cb77e | ||
|
|
a3b79215c4 | ||
|
|
0cbe76329e | ||
|
|
c93eef06f6 | ||
|
|
60f5662d81 | ||
|
|
6531413325 | ||
|
|
fce39548d0 | ||
|
|
0f26e7d060 | ||
|
|
c98df6dd97 | ||
|
|
7a24fd0367 | ||
|
|
30d177165d | ||
|
|
e4762a1a70 | ||
|
|
e0356bc9c5 | ||
|
|
aa1f2bc0f6 | ||
|
|
0272c7b9ed | ||
|
|
10ec823151 | ||
|
|
fdfc6bc997 | ||
|
|
3f1677bff2 | ||
|
|
782bc4b78a | ||
|
|
f2763d5af5 | ||
|
|
573d6c75ca | ||
|
|
42d5f4baf1 | ||
|
|
8320571c4d | ||
|
|
559e71205d | ||
|
|
1bac4352e3 | ||
|
|
b64ab9b0b0 | ||
|
|
049fe5b68b | ||
|
|
b716d077c4 | ||
|
|
09fa8afefe | ||
|
|
129239a742 | ||
|
|
ee53c54255 | ||
|
|
f4d212ae18 | ||
|
|
e2f7991ad8 | ||
|
|
c0c455358f | ||
|
|
488ddd4bcb | ||
|
|
b2c1ae0068 | ||
|
|
21ce3a2242 | ||
|
|
6fa606ffd5 | ||
|
|
c1c7cba96a | ||
|
|
4992a3cb76 | ||
|
|
bbac863a2a | ||
|
|
44399b1984 | ||
|
|
4288ceae56 | ||
|
|
38080aff92 | ||
|
|
9999a90e62 | ||
|
|
7892ec6006 | ||
|
|
362818530a | ||
|
|
643cea4930 | ||
|
|
b271a898f5 | ||
|
|
35545facce | ||
|
|
170602e85f | ||
|
|
c6e6b54b8f | ||
|
|
9dba3860e2 | ||
|
|
a187ba98f1 | ||
|
|
203e923f99 | ||
|
|
ba78dc2d27 | ||
|
|
b55eb23edd | ||
|
|
e2dc330cf9 | ||
|
|
de75ace988 | ||
|
|
3d48ecac37 | ||
|
|
a27f72eab9 | ||
|
|
7c0e25f253 | ||
|
|
3ca197b45c | ||
|
|
48edb03b32 | ||
|
|
40d9aee6e6 | ||
|
|
2fb4df6bdd | ||
|
|
4c75db9a95 | ||
|
|
18a511b1c9 | ||
|
|
15f73f9442 | ||
|
|
bfd7eb2141 | ||
|
|
803584dc7c | ||
|
|
ac136dab08 | ||
|
|
f31533d8d6 | ||
|
|
8c93e287ff | ||
|
|
846123c57a | ||
|
|
694233e2f0 | ||
|
|
34b58e41c4 | ||
|
|
a6a1272d17 | ||
|
|
60c1549168 | ||
|
|
c3f10c507d | ||
|
|
230cc467a1 | ||
|
|
084c1ec5e5 | ||
|
|
150fcda6d3 | ||
|
|
dd9547bbc1 | ||
|
|
acb5112f03 | ||
|
|
53a6ea0f8a | ||
|
|
4efa9a38ef | ||
|
|
de6fe93b75 | ||
|
|
37fe8b21bc | ||
|
|
7fceefbe5c | ||
|
|
434d3877d7 | ||
|
|
bf34a23e68 | ||
|
|
ae62d2b474 | ||
|
|
bb72885027 | ||
|
|
d24ee428f3 | ||
|
|
cbb3b3e90b | ||
|
|
b531747918 | ||
|
|
669fee5bc8 | ||
|
|
e7ef940e24 | ||
|
|
4e124091b4 | ||
|
|
a3ef215485 | ||
|
|
b07ac78d68 | ||
|
|
544992be88 | ||
|
|
6c2d8de53f | ||
|
|
c7e7d47c4e | ||
|
|
2c6bf85f7f | ||
|
|
bab4ce4bd5 | ||
|
|
2f55f423b9 | ||
|
|
1b7ade9317 | ||
|
|
747383c847 | ||
|
|
8764759323 | ||
|
|
5b04aa28c5 | ||
|
|
3b72fefe23 | ||
|
|
508158112d | ||
|
|
d1e9617eae | ||
|
|
6c88c0ba3c | ||
|
|
a5bdf4d13c | ||
|
|
eb680a204e | ||
|
|
e27298e444 | ||
|
|
f69ee7e8de | ||
|
|
04e7e5b3ab | ||
|
|
5c58e46417 | ||
|
|
582c331117 | ||
|
|
bffcc67fa4 | ||
|
|
7a7a01eeaa | ||
|
|
0f0d24b510 | ||
|
|
1f37b46151 | ||
|
|
225c0db092 | ||
|
|
9c29cb29ff | ||
|
|
51e64e1891 | ||
|
|
2f159ac912 | ||
|
|
c9867097e8 | ||
|
|
fc983927ce | ||
|
|
2f4792253a | ||
|
|
2149a80852 | ||
|
|
9b611bb7ff | ||
|
|
52ed86284a | ||
|
|
d64177bde5 | ||
|
|
35f4c674e9 | ||
|
|
1291b328d0 | ||
|
|
ac947b1543 | ||
|
|
9b9ec47bcf | ||
|
|
2e34a9f4ea | ||
|
|
d0193d3c10 | ||
|
|
39e7c45b4f | ||
|
|
1c499ac8a7 | ||
|
|
45d1230102 | ||
|
|
e70e3713fc | ||
|
|
9c40ff4879 | ||
|
|
b55eeef7a3 | ||
|
|
2e8157047d | ||
|
|
1e44f82eb5 | ||
|
|
ba2e8ad583 | ||
|
|
b55b71ff0a | ||
|
|
9da2ccc812 | ||
|
|
f46dbc5ca7 | ||
|
|
002285a5af | ||
|
|
bcbaae5ef5 | ||
|
|
4a7d46dd1f | ||
|
|
7d05220cfa | ||
|
|
18fcddfc34 | ||
|
|
1fe526a734 | ||
|
|
737baa9d0e | ||
|
|
b3fe7c1436 | ||
|
|
e36ad64aa6 | ||
|
|
d45e1dff0f | ||
|
|
10b2a17718 | ||
|
|
caca08fd5b | ||
|
|
7d56daacca | ||
|
|
440b86633a | ||
|
|
5e9bdba777 | ||
|
|
2c6b89751d | ||
|
|
872cb1d006 | ||
|
|
0115424167 | ||
|
|
f9811f0c59 | ||
|
|
8334978a43 | ||
|
|
bf51f725e2 | ||
|
|
7fc411bdd7 | ||
|
|
ebd8f7ff47 | ||
|
|
a9d33a8873 | ||
|
|
11e1a93c59 | ||
|
|
a6a69a16f7 | ||
|
|
756ed95b43 | ||
|
|
ac09fc1abd | ||
|
|
75830df509 | ||
|
|
654a1cb67a | ||
|
|
87f3e731b1 | ||
|
|
16a9708790 | ||
|
|
ad1b0f9880 | ||
|
|
44ffd340f5 | ||
|
|
aa4ebf5f47 | ||
|
|
bf1517a60e | ||
|
|
1d6ae7b376 | ||
|
|
810e95fe11 | ||
|
|
aa5d8e094a | ||
|
|
686cbf7272 | ||
|
|
2f99103ed1 | ||
|
|
17dbbce0d2 | ||
|
|
664e5e2644 | ||
|
|
fd98efc95c | ||
|
|
2e955bfe57 | ||
|
|
4c711d790a | ||
|
|
d187c66987 | ||
|
|
c7ec650cfc | ||
|
|
2c463ec517 | ||
|
|
ba336f2884 | ||
|
|
c2088bff6c | ||
|
|
5164c24342 | ||
|
|
e679e93362 | ||
|
|
e7972ac1b4 | ||
|
|
290537c581 | ||
|
|
3ec884f881 | ||
|
|
19e73a3ebd | ||
|
|
23c9597870 | ||
|
|
2d5e89d545 | ||
|
|
445f66ae1e | ||
|
|
e7737ee7af | ||
|
|
93c8ad75a6 | ||
|
|
6ce63cad2c | ||
|
|
f14e95ec9d | ||
|
|
84a41cba7d | ||
|
|
e83fa1d76a | ||
|
|
3d88db6f21 | ||
|
|
7fcb9d97d3 | ||
|
|
75bf60667d | ||
|
|
077fc3f23a | ||
|
|
6748d86dae | ||
|
|
bc66ec5aee | ||
|
|
c8d4f58f0d | ||
|
|
3910df215a | ||
|
|
e66e7dc6d5 | ||
|
|
1710e07231 | ||
|
|
2efe05def2 | ||
|
|
dffedb115e | ||
|
|
6203a88615 | ||
|
|
54c2f0960a | ||
|
|
bb6750a982 | ||
|
|
1310591d05 | ||
|
|
8cbb7f3834 | ||
|
|
858d157863 |
48 changed files with 3869 additions and 955 deletions
|
|
@ -1,2 +1 @@
|
|||
{:lint-as
|
||||
{honeysql.helpers/defhelper clojure.core/defn}}
|
||||
{}
|
||||
|
|
|
|||
1
.clj-kondo/imports/babashka/fs/config.edn
Normal file
1
.clj-kondo/imports/babashka/fs/config.edn
Normal file
|
|
@ -0,0 +1 @@
|
|||
{:lint-as {babashka.fs/with-temp-dir clojure.core/let}}
|
||||
3
.clj-kondo/imports/http-kit/http-kit/config.edn
Normal file
3
.clj-kondo/imports/http-kit/http-kit/config.edn
Normal file
|
|
@ -0,0 +1,3 @@
|
|||
|
||||
{:hooks
|
||||
{:analyze-call {org.httpkit.server/with-channel httpkit.with-channel/with-channel}}}
|
||||
|
|
@ -0,0 +1,16 @@
|
|||
(ns httpkit.with-channel
|
||||
(:require [clj-kondo.hooks-api :as api]))
|
||||
|
||||
(defn with-channel [{node :node}]
|
||||
(let [[request channel & body] (rest (:children node))]
|
||||
(when-not (and request channel) (throw (ex-info "No request or channel provided" {})))
|
||||
(when-not (api/token-node? channel) (throw (ex-info "Missing channel argument" {})))
|
||||
(let [new-node
|
||||
(api/list-node
|
||||
(list*
|
||||
(api/token-node 'let)
|
||||
(api/vector-node [channel (api/vector-node [])])
|
||||
request
|
||||
body))]
|
||||
|
||||
{:node new-node})))
|
||||
5
.clj-kondo/imports/rewrite-clj/rewrite-clj/config.edn
Normal file
5
.clj-kondo/imports/rewrite-clj/rewrite-clj/config.edn
Normal file
|
|
@ -0,0 +1,5 @@
|
|||
{:lint-as
|
||||
{rewrite-clj.zip/subedit-> clojure.core/->
|
||||
rewrite-clj.zip/subedit->> clojure.core/->>
|
||||
rewrite-clj.zip/edit-> clojure.core/->
|
||||
rewrite-clj.zip/edit->> clojure.core/->>}}
|
||||
1
.clj-kondo/imports/taoensso/encore/config.edn
Normal file
1
.clj-kondo/imports/taoensso/encore/config.edn
Normal file
|
|
@ -0,0 +1 @@
|
|||
{:hooks {:analyze-call {taoensso.encore/defalias taoensso.encore/defalias}}}
|
||||
16
.clj-kondo/imports/taoensso/encore/taoensso/encore.clj
Normal file
16
.clj-kondo/imports/taoensso/encore/taoensso/encore.clj
Normal file
|
|
@ -0,0 +1,16 @@
|
|||
(ns taoensso.encore
|
||||
(:require
|
||||
[clj-kondo.hooks-api :as hooks]))
|
||||
|
||||
(defn defalias [{:keys [node]}]
|
||||
(let [[sym-raw src-raw] (rest (:children node))
|
||||
src (if src-raw src-raw sym-raw)
|
||||
sym (if src-raw
|
||||
sym-raw
|
||||
(symbol (name (hooks/sexpr src))))]
|
||||
{:node (with-meta
|
||||
(hooks/list-node
|
||||
[(hooks/token-node 'def)
|
||||
(hooks/token-node (hooks/sexpr sym))
|
||||
(hooks/token-node (hooks/sexpr src))])
|
||||
(meta src))}))
|
||||
10
.github/workflows/test-and-release.yml
vendored
10
.github/workflows/test-and-release.yml
vendored
|
|
@ -9,19 +9,19 @@ jobs:
|
|||
build-and-release:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v3
|
||||
- uses: actions/checkout@v4
|
||||
with:
|
||||
fetch-depth: 0
|
||||
- uses: actions/setup-java@v3
|
||||
- uses: actions/setup-java@v4
|
||||
with:
|
||||
distribution: 'adopt'
|
||||
distribution: 'temurin'
|
||||
java-version: '11'
|
||||
- name: Setup Clojure
|
||||
uses: DeLaGuardo/setup-clojure@master
|
||||
with:
|
||||
cli: '1.11.1.1208'
|
||||
cli: '1.12.0.1530'
|
||||
- name: Cache All The Things
|
||||
uses: actions/cache@v3
|
||||
uses: actions/cache@v4
|
||||
with:
|
||||
path: |
|
||||
~/.m2/repository
|
||||
|
|
|
|||
22
.github/workflows/test-and-snapshot.yml
vendored
22
.github/workflows/test-and-snapshot.yml
vendored
|
|
@ -9,17 +9,17 @@ jobs:
|
|||
build-and-snapshot:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v3
|
||||
- uses: actions/setup-java@v3
|
||||
- uses: actions/checkout@v4
|
||||
- uses: actions/setup-java@v4
|
||||
with:
|
||||
distribution: 'adopt'
|
||||
distribution: 'temurin'
|
||||
java-version: '11'
|
||||
- name: Setup Clojure
|
||||
uses: DeLaGuardo/setup-clojure@master
|
||||
with:
|
||||
cli: '1.11.1.1208'
|
||||
cli: '1.12.0.1530'
|
||||
- name: Cache All The Things
|
||||
uses: actions/cache@v3
|
||||
uses: actions/cache@v4
|
||||
with:
|
||||
path: |
|
||||
~/.m2/repository
|
||||
|
|
@ -39,19 +39,19 @@ jobs:
|
|||
runs-on: ubuntu-latest
|
||||
strategy:
|
||||
matrix:
|
||||
java: [ '8', '14', '17', '19' ]
|
||||
java: [ '8', '17', '21' ]
|
||||
steps:
|
||||
- uses: actions/checkout@v3
|
||||
- uses: actions/setup-java@v3
|
||||
- uses: actions/checkout@v4
|
||||
- uses: actions/setup-java@v4
|
||||
with:
|
||||
distribution: 'adopt'
|
||||
distribution: 'temurin'
|
||||
java-version: ${{ matrix.java }}
|
||||
- name: Clojure CLI
|
||||
uses: DeLaGuardo/setup-clojure@master
|
||||
with:
|
||||
cli: '1.11.1.1208'
|
||||
cli: '1.12.0.1530'
|
||||
- name: Cache All The Things
|
||||
uses: actions/cache@v3
|
||||
uses: actions/cache@v4
|
||||
with:
|
||||
path: |
|
||||
~/.m2/repository
|
||||
|
|
|
|||
33
.github/workflows/test-bb.yml
vendored
Normal file
33
.github/workflows/test-bb.yml
vendored
Normal file
|
|
@ -0,0 +1,33 @@
|
|||
name: Babashka tests
|
||||
|
||||
on:
|
||||
pull_request:
|
||||
push:
|
||||
branches:
|
||||
- develop
|
||||
|
||||
jobs:
|
||||
build:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
- uses: actions/setup-java@v4
|
||||
with:
|
||||
distribution: 'temurin'
|
||||
java-version: 21
|
||||
- name: Clojure CLI
|
||||
uses: DeLaGuardo/setup-clojure@master
|
||||
with:
|
||||
cli: '1.12.0.1530'
|
||||
bb: latest
|
||||
- name: Cache All The Things
|
||||
uses: actions/cache@v4
|
||||
with:
|
||||
path: |
|
||||
~/.m2/repository
|
||||
~/.gitlibs
|
||||
~/.clojure
|
||||
~/.cpcache
|
||||
key: ${{ runner.os }}-${{ hashFiles('**/deps.edn', '**/bb.edn') }}
|
||||
- name: Run Tests
|
||||
run: bb test
|
||||
12
.github/workflows/test.yml
vendored
12
.github/workflows/test.yml
vendored
|
|
@ -7,19 +7,19 @@ jobs:
|
|||
runs-on: ubuntu-latest
|
||||
strategy:
|
||||
matrix:
|
||||
java: [ '8', '11', '14', '17', '19' ]
|
||||
java: [ '8', '11', '17', '21' ]
|
||||
steps:
|
||||
- uses: actions/checkout@v3
|
||||
- uses: actions/setup-java@v3
|
||||
- uses: actions/checkout@v4
|
||||
- uses: actions/setup-java@v4
|
||||
with:
|
||||
distribution: 'adopt'
|
||||
distribution: 'temurin'
|
||||
java-version: ${{ matrix.java }}
|
||||
- name: Clojure CLI
|
||||
uses: DeLaGuardo/setup-clojure@master
|
||||
with:
|
||||
cli: '1.11.1.1208'
|
||||
cli: '1.12.0.1530'
|
||||
- name: Cache All The Things
|
||||
uses: actions/cache@v3
|
||||
uses: actions/cache@v4
|
||||
with:
|
||||
path: |
|
||||
~/.m2/repository
|
||||
|
|
|
|||
10
.gitignore
vendored
10
.gitignore
vendored
|
|
@ -1,4 +1,9 @@
|
|||
*.class
|
||||
*.jar
|
||||
*.swp
|
||||
*~
|
||||
.calva/output-window/
|
||||
.calva/repl.calva-repl
|
||||
.classpath
|
||||
.clj-kondo/.cache
|
||||
.cpcache
|
||||
|
|
@ -18,11 +23,6 @@
|
|||
.settings
|
||||
.socket-repl-port
|
||||
.sw*
|
||||
.vscode
|
||||
*.class
|
||||
*.jar
|
||||
*.swp
|
||||
*~
|
||||
/checkouts
|
||||
/classes
|
||||
/cljs-test-runner-out
|
||||
|
|
|
|||
23
.gitpod.yml
23
.gitpod.yml
|
|
@ -1,23 +0,0 @@
|
|||
image:
|
||||
file: .gitpod.dockerfile
|
||||
|
||||
vscode:
|
||||
extensions:
|
||||
- betterthantomorrow.calva
|
||||
- mauricioszabo.clover
|
||||
|
||||
tasks:
|
||||
- name: Prepare deps/clover
|
||||
init: |
|
||||
clojure -A:test -P
|
||||
echo 50505 > .socket-repl-port
|
||||
mkdir ~/.config/clover
|
||||
cp .clover/config.cljs ~/.config/clover/
|
||||
- name: Start REPL
|
||||
command: clojure -J-Dclojure.server.repl="{:address \"0.0.0.0\" :port 50505 :accept clojure.core.server/repl}" -A:test
|
||||
- name: See Changes
|
||||
command: code CHANGELOG.md
|
||||
|
||||
github:
|
||||
prebuilds:
|
||||
develop: true
|
||||
140
CHANGELOG.md
140
CHANGELOG.md
|
|
@ -1,5 +1,141 @@
|
|||
# Changes
|
||||
|
||||
* 2.7.next in progress
|
||||
* Address [#440](https://github.com/seancorfield/honeysql/issues/440) by supporting multiple tables in `:truncate`.
|
||||
* Support `USING HASH` as well as `USING GIN`.
|
||||
* Fix [#571](https://github.com/seancorfield/honeysql/issues/571) by allowing `:order-by` to take an empty sequence of columns (and be omitted).
|
||||
* Update dev/build deps.
|
||||
|
||||
* 2.7.1295 -- 2025-03-12
|
||||
* Address #570 by adding `:.:.` as special syntax for Snowflake's JSON path syntax, and `:at` as special syntax for general `[`..`]` path syntax.
|
||||
* Drop support for Clojure 1.9 [#561](https://github.com/seancorfield/honeysql/issues/561).
|
||||
|
||||
* 2.6.1281 -- 2025-03-06
|
||||
* Address [#568](https://github.com/seancorfield/honeysql/issues/568) by adding `honey.sql/semicolon` to merge multiple SQL+params vectors into one (with semicolons separating the SQL statements).
|
||||
* Address [#567](https://github.com/seancorfield/honeysql/issues/567) by adding support for `ASSERT` clause.
|
||||
* Address [#566](https://github.com/seancorfield/honeysql/issues/566) by adding `IS [NOT] DISTINCT FROM` operators.
|
||||
* Add examples of `:alias` with `:group-by` (syntax is slightly different to existing examples for `:order-by`).
|
||||
|
||||
* 2.6.1270 -- 2025-01-17
|
||||
* Fix autoboxing introduced in 2.6.1267 via PR [#564](https://github.com/seancorfield/honeysql/pull/564) [@alexander-yakushev](https://github.com/alexander-yakushev).
|
||||
|
||||
* 2.6.1267 -- 2025-01-16
|
||||
* Support expressions in `WITH` clauses via PR [#563](https://github.com/seancorfield/honeysql/pull/563) [@krevedkokun](https://github.com/krevedkokun).
|
||||
* More performance optimizations via PRs [#560](https://github.com/seancorfield/honeysql/pull/560) and [#562](https://github.com/seancorfield/honeysql/pull/562) [@alexander-yakushev](https://github.com/alexander-yakushev).
|
||||
* Fix two broken links to the [HoneySQL web app](https://john.shaffe.rs/honeysql/) via PR [#559](https://github.com/seancorfield/honeysql/pull/559) [@whatacold](https://github.com/whatacold).
|
||||
* Make SQL Server dialect auto-lift Boolean values to parameters since SQL Server has no `TRUE` / `FALSE` literals.
|
||||
* Fix bug in `DEFAULT` values clause (that omitted some values).
|
||||
|
||||
* 2.6.1243 -- 2024-12-13
|
||||
* Address [#558](https://github.com/seancorfield/honeysql/issues/558) by adding `:patch-into` (and `patch-into` helper) for XTDB (but in core).
|
||||
* Address [#556](https://github.com/seancorfield/honeysql/issues/556) by adding an XTDB section to the documentation with examples.
|
||||
* Address [#555](https://github.com/seancorfield/honeysql/issues/555) by supporting `SETTING` clause for XTDB.
|
||||
* Replace `assert` calls with proper validation, throwing `ex-info` on failure (like other existing validation in HoneySQL).
|
||||
* Experimental `:xtdb` dialect removed (since XTDB no longer supports qualified column names).
|
||||
* Update dev/test deps.
|
||||
|
||||
* 2.6.1230 -- 2024-11-23
|
||||
* Fix [#553](https://github.com/seancorfield/honeysql/issues/553) by adding `:not-between` as special syntax via PR [#554](https://github.com/seancorfield/honeysql/pull/554) [@plooney81](https://github.com/plooney81)
|
||||
* Fix [#552](https://github.com/seancorfield/honeysql/issues/552) by changing the assert-on-load behavior into an explicit test in the test suite.
|
||||
* Fix [#551](https://github.com/seancorfield/honeysql/issues/551) by supporting multiple `WINDOW` clauses.
|
||||
* Fix [#549](https://github.com/seancorfield/honeysql/issues/549) by using `:bb` conditionals to support Babashka (and still support Clojure 1.9.0), and add testing against Babashka so it is fully-supported as a target via PR [#550](https://github.com/seancorfield/honeysql/pull/550) [@borkdude](https://github.com/borkdude)
|
||||
* Address [#532](https://github.com/seancorfield/honeysql/issues/532) by adding support for XTDB SQL extensions `ERASE`, `EXCLUDE`, `OBJECT`, `RECORD`, `RECORDS`, and `RENAME`, along with inline hash maps (as records) and `:get-in` for object navigation, and starting to write tests for XTDB compatibility.
|
||||
|
||||
* 2.6.1203 -- 2024-10-22
|
||||
* Fix [#548](https://github.com/seancorfield/honeysql/issues/548) which was a regression introduced in [#526](https://github.com/seancorfield/honeysql/issues/526) (in 2.6.1161).
|
||||
* Address [#542](https://github.com/seancorfield/honeysql/issues/542) by adding support for `WITH` query tail options for PostgreSQL.
|
||||
* Replace all optional argument destructuring with multiple arities to improve performance.
|
||||
|
||||
* 2.6.1196 -- 2024-10-06
|
||||
* Address [#547](https://github.com/seancorfield/honeysql/issues/547) by adding examples of conditional SQL building with the helpers to the README and the `honey.sql.helpers` ns docstring.
|
||||
* Performance optimizations via PRs [#545](https://github.com/seancorfield/honeysql/pull/545) and [#546](https://github.com/seancorfield/honeysql/pull/546) [@alexander-yakushev](https://github.com/alexander-yakushev).
|
||||
* Address [#544](https://github.com/seancorfield/honeysql/issues/544) by adding support for MySQL's `VALUES ROW(..)` syntax.
|
||||
* Fix [#543](https://github.com/seancorfield/honeysql/issues/543) by supporting both symbols and keywords in named parameters.
|
||||
* Address [#541](https://github.com/seancorfield/honeysql/issues/541) by specifying the expected result of a formatter function passed to `register-clause!` and adding the example from the README to **Extending HoneySQL**.
|
||||
* Getting Started updated based on feedback from Los Angeles Clojure meetup walkthrough [#539](https://github.com/seancorfield/honeysql/issues/539).
|
||||
* Fix [#538](https://github.com/seancorfield/honeysql/issues/538) by removing `mod` from list of infix operators.
|
||||
* Fixed a few symbol/keyword resolution bugs in the formatter. Thanks to [@irigarae](https://github.com/irigarae).
|
||||
* Update Clojure version to 1.12.0; update dev/test/ci deps.
|
||||
|
||||
* 2.6.1161 -- 2024-08-29
|
||||
* Address [#537](https://github.com/seancorfield/honeysql/issues/537) by ignoring non-scalar values in metadata, and expanding support to numbers, and checking strings for suspicious characters.
|
||||
* Address [#536](https://github.com/seancorfield/honeysql/issues/536) by noting what will not work with PostgreSQL (but works with other databases).
|
||||
* Address [#533](https://github.com/seancorfield/honeysql/issues/533) by adding `honey.sql/*escape-?*` which can be bound to `false` to prevent `?` being escaped to `??` when used as an operator or function.
|
||||
* Address [#526](https://github.com/seancorfield/honeysql/issues/526) by using `format-var` in DDL, instead of `format-entity`.
|
||||
* Update JDK test matrix (adopt -> temurin, 19 -> 21).
|
||||
* Update Clojure versions (to 1.11.4 & 1.12.0-rc2).
|
||||
|
||||
* 2.6.1147 -- 2024-06-12
|
||||
* Address [#531](https://github.com/seancorfield/honeysql/issues/531) and [#527](https://github.com/seancorfield/honeysql/issues/527) by adding tests and more documentation for `:composite`; fix bug in `set-dialect!` where clause order is not restored.
|
||||
* Address [#530](https://github.com/seancorfield/honeysql/issues/530) by adding support for `:using-gin` to `:create-index`.
|
||||
* Address [#529](https://github.com/seancorfield/honeysql/issues/529) by fixing `:join` special syntax to support aliases and to handle expressions the same way `select` / `from` etc handle them (extra `[...]` nesting).
|
||||
* Add example of mixed `DO UPDATE SET` with `EXCLUDED` and regular SQL expressions.
|
||||
* Improve exception message when un-`lift`-ed JSON expressions are used in the DSL.
|
||||
* Update Clojure versions (to 1.11.3 and 1.12.0-alpha12); update other dev/test dependencies.
|
||||
|
||||
* 2.6.1126 -- 2024-03-04
|
||||
* Address [#524](https://github.com/seancorfield/honeysql/issues/524) by adding example of `{:nest ..}` in `:union` clause reference docs.
|
||||
* Address [#523](https://github.com/seancorfield/honeysql/issues/523) by expanding examples in README **Functions** to show aliases.
|
||||
* Address [#522](https://github.com/seancorfield/honeysql/issues/522) by supporting metadata on table specifications in `:from` and `:join` clauses to provide index hints (SQL Server).
|
||||
* ~Address [#521](https://github.com/seancorfield/honeysql/issues/521) by adding initial experimental support for an XTDB dialect.~ _[This was removed in 2.6.1243 since XTDB no longer supports qualified column names]_
|
||||
* Address [#520](https://github.com/seancorfield/honeysql/issues/520) by expanding how `:inline` works, to support a sequence of arguments.
|
||||
* Fix [#518](https://github.com/seancorfield/honeysql/issues/518) by moving temporal clause before alias.
|
||||
* Address [#495](https://github.com/seancorfield/honeysql/issues/495) by adding `formatv` macro (`.clj` only!) -- and removing the experimental `formatf` function (added for discussion in 2.4.1045).
|
||||
* Implemented `CREATE INDEX` [#348](https://github.com/seancorfield/honeysql/issues/348) via PR [#517](https://github.com/seancorfield/honeysql/pull/517) [@dancek](https://github.com/dancek).
|
||||
* Mention `:not-in` explicitly in the documentation.
|
||||
* Code cleanup per `clj-kondo`.
|
||||
|
||||
* 2.5.1103 -- 2023-12-03
|
||||
* Address [#515](https://github.com/seancorfield/honeysql/issues/515) by:
|
||||
* Quoting entities that start with a digit but are otherwise alphanumeric. Note that entities that are all digits (optionally including underscores) will still not be quoted as in previous releases,
|
||||
* Adding a new `:quoted-always` option allows users to specify a regex that matches entities that should always be quoted (stropped) regardless of the value of `:quoted` (such as reserved words that you have used as column or table names).
|
||||
* Address [#513](https://github.com/seancorfield/honeysql/issues/513) by:
|
||||
* Ignoring `:file`, `:line`, `:column`, `:end-line`, and `:end-column` metadata keys (previously only `:line` and `:column` were ignored),
|
||||
* Adding an `:ignored-metadata` option to allow additional keys to be ignored.
|
||||
|
||||
* 2.5.1091 -- 2023-10-28
|
||||
* Address [#512](https://github.com/seancorfield/honeysql/issues/512) by adding support for subqueries in the `:array` special syntax (for BigQuery and PostgreSQL). This also adds support for metadata on the `:select` value to produce `AS STRUCT` or `DISTINCT`.
|
||||
* Address [#511](https://github.com/seancorfield/honeysql/issues/511) by adding support for BigQuery `CREATE OR REPLACE`.
|
||||
* Address [#510](https://github.com/seancorfield/honeysql/issues/510) by adding initial support for an NRQL dialect.
|
||||
* Fix [#509](https://github.com/seancorfield/honeysql/issues/509) by checking for `ident?` before checking keyword/symbol.
|
||||
|
||||
* 2.4.1078 -- 2023-10-07
|
||||
* Address [#507](https://github.com/seancorfield/honeysql/issues/507) by clarifying formatting of `:cast` in **Special Syntax**.
|
||||
* Fix [#505](https://github.com/seancorfield/honeysql/issues/505) by rewriting the helper merge function to handle both keywords and symbols properly.
|
||||
* Address [#503](https://github.com/seancorfield/honeysql/issues/503) by adding `:at-time-zone` special syntax.
|
||||
* Address [#504](https://github.com/seancorfield/honeysql/issues/504) for BigQuery support, by adding special syntax for ignore/respect nulls, as well as new `:distinct` and `:expr` clauses to allow expressions to be qualified with SQL clauses. The latter will probably be useful for other dialects too.
|
||||
* Update `tools.build` to 0.9.6 (and get rid of `template/pom.xml` in favor of new `:pom-data` option to `b/write-pom`).
|
||||
|
||||
* 2.4.1066 -- 2023-08-27
|
||||
* Add `:select` with function call and alias example to README (PR [#502](https://github.com/seancorfield/honeysql/pull/502) [@markbastian](https://github.com/markbastian)).
|
||||
* Address [#501](https://github.com/seancorfield/honeysql/issues/501) by making `INSERT INTO` (and `REPLACE INTO`) use the `:columns` or `:values` clauses to produce column names (which are then omitted from those other clauses).
|
||||
* Address [#497](https://github.com/seancorfield/honeysql/issues/497) by adding `:alias` special syntax.
|
||||
* Address [#496](https://github.com/seancorfield/honeysql/issues/496) by adding `:overriding-value` option to `:insert` clause.
|
||||
* Address [#407](https://github.com/seancorfield/honeysql/issues/407) by adding support for temporal queries (see `FROM` in [SQL Clause Reference](https://cljdoc.org/d/com.github.seancorfield/honeysql/CURRENT/doc/getting-started/sql-clause-reference#from)).
|
||||
* Address [#389](https://github.com/seancorfield/honeysql/issues/389) by adding examples of `[:only :table]` producing `ONLY(table)`.
|
||||
* Add `:create-or-replace-view` to support PostgreSQL's lack of `IF NOT EXISTS` for `CREATE VIEW`.
|
||||
* Attempt to clarify the formatting behavior of the `:values` clause when used to produce column names.
|
||||
* Update `tools.build` to 0.9.5 (and remove `:java-opts` setting from `build/run-task`)
|
||||
|
||||
* 2.4.1045 -- 2023-06-25
|
||||
* Address [#495](https://github.com/seancorfield/honeysql/issues/495) by adding (experimental) `formatf` function -- _note: this was removed in 2.6.1126, in favor of the `formatv` macro._
|
||||
* Fix [#494](https://github.com/seancorfield/honeysql/issues/494) by supporting expressions in `:on-conflict` instead of just entities.
|
||||
* Address [#493](https://github.com/seancorfield/honeysql/issues/493) by clarifying use of `:values` in CTEs (using `:with`).
|
||||
* Address [#489](https://github.com/seancorfield/honeysql/issues/489) by adding more examples around `:update`.
|
||||
* Attempt to improve `honey.sql.helpers` namespace docstring (by adding a note from the relevant **Getting Started** section).
|
||||
* Update dev/test dependencies.
|
||||
|
||||
* 2.4.1033 -- 2023-05-22
|
||||
* Tentative [ClojureCLR](https://github.com/clojure/clojure-clr) support.
|
||||
* Improve `on-conflict` helper docstring [#490](https://github.com/seancorfield/honeysql/pull/490) [@holyjak](https://github.com/holyjak).
|
||||
|
||||
* 2.4.1026 -- 2023-04-15
|
||||
* Fix [#486](https://github.com/seancorfield/honeysql/issues/486) by supporting ANSI-style `INTERVAL` syntax.
|
||||
* Fix [#485](https://github.com/seancorfield/honeysql/issues/485) by adding `:with-ordinality` "operator".
|
||||
* Fix [#484](https://github.com/seancorfield/honeysql/issues/484) by adding `TABLE` to `TRUNCATE`.
|
||||
* Fix [#483](https://github.com/seancorfield/honeysql/issues/483) by adding a function-like `:join` syntax to produce nested `JOIN` expressions.
|
||||
* Update `tools.build`; split alias `:test`/`:runner` for friendlier jack-in UX while developing.
|
||||
|
||||
* 2.4.1011 -- 2023-03-23
|
||||
* Address [#481](https://github.com/seancorfield/honeysql/issues/481) by adding more examples around `:do-update-set`.
|
||||
* Address [#480](https://github.com/seancorfield/honeysql/issues/480) by clarifying the general relationship between clauses and helpers.
|
||||
|
|
@ -39,7 +175,7 @@
|
|||
|
||||
* 2.4.962 -- 2022-12-17
|
||||
* Fix `set-options!` (only `:checking` worked in 2.4.947).
|
||||
* Fix `:cast` formatting when quoting is enabled, via PR [#443](https://github.com/seancorfield/honeysql/pull/443) [duddlf23](https://github.com/duddlf23).
|
||||
* Fix `:cast` formatting when quoting is enabled, via PR [#443](https://github.com/seancorfield/honeysql/pull/443) [duddlf23](https://github.com/duddlf23). **This changes how type names containing `-` are formatted in a cast.** See [`cast` Special Syntax](https://cljdoc.org/d/com.github.seancorfield/honeysql/CURRENT/doc/getting-started/sql-special-syntax-#cast) for more details.
|
||||
* Fix [#441](https://github.com/seancorfield/honeysql/issues/441) by adding `:replace-into` to in-flight clause order (as well as registering it for the `:mysql` dialect).
|
||||
* Fix [#434](https://github.com/seancorfield/honeysql/issues/434) by special-casing `:'ARRAY`.
|
||||
* Fix [#433](https://github.com/seancorfield/honeysql/issues/433) by supporting additional `WITH` syntax, via PR [#432](https://github.com/seancorfield/honeysql/issues/432), [@MawiraIke](https://github.com/MawiraIke). _[Technically, this was in 2.4.947, but I kept the issue open while I wordsmithed the documentation]_
|
||||
|
|
@ -140,7 +276,7 @@
|
|||
* Fixes #344 by no longer dropping the qualifier on columns in a `SET` clause _for the `:mysql` dialect only_; the behavior is unchanged for all other dialects.
|
||||
* Fixes #340 by making the "hyphen to space" logic more general so _operators_ containing `-` should retain the hyphen without special cases.
|
||||
* Documentation improvements: `:fetch`, `:lift`, `:limit`, `:offset`, `:param`, `:select`; also around JSON/PostgreSQL.
|
||||
* Link to the [HoneySQL web app](https://www.john-shaffer.com/honeysql/) in both the README and **Getting Started**.
|
||||
* Link to the [HoneySQL web app](https://john.shaffe.rs/honeysql/) in both the README and **Getting Started**.
|
||||
* Switch to `tools.build` for running tests and JAR building etc.
|
||||
|
||||
* 2.0.0-rc5 (for testing; 2021-07-17)
|
||||
|
|
|
|||
175
README.md
175
README.md
|
|
@ -1,18 +1,24 @@
|
|||
# Honey SQL [](https://github.com/seancorfield/honeysql/actions/workflows/test.yml) [](https://gitpod.io/#https://github.com/seancorfield/honeysql)
|
||||
# Honey SQL [](https://github.com/seancorfield/honeysql/actions/workflows/test-and-release.yml) [](https://github.com/seancorfield/honeysql/actions/workflows/test-and-snapshot.yml) [](https://github.com/seancorfield/honeysql/actions/workflows/test.yml)
|
||||
|
||||
SQL as Clojure data structures. Build queries programmatically -- even at runtime -- without having to bash strings together.
|
||||
|
||||
## Build
|
||||
|
||||
[](https://clojars.org/com.github.seancorfield/honeysql) [](https://cljdoc.org/d/com.github.seancorfield/honeysql/CURRENT)
|
||||
[](https://clojars.org/com.github.seancorfield/honeysql)
|
||||
[](https://cljdoc.org/d/com.github.seancorfield/honeysql/CURRENT)
|
||||
[](https://clojurians.slack.com/app_redirect?channel=honeysql)
|
||||
[](http://clojurians.net)
|
||||
[](https://clojurians.zulipchat.com/#narrow/channel/152091-honeysql)
|
||||
|
||||
This project follows the version scheme MAJOR.MINOR.COMMITS where MAJOR and MINOR provide some relative indication of the size of the change, but do not follow semantic versioning. In general, all changes endeavor to be non-breaking (by moving to new names rather than by breaking existing names). COMMITS is an ever-increasing counter of commits since the beginning of this repository.
|
||||
|
||||
> Note: every commit to the **develop** branch runs CI (GitHub Actions) and successful runs push a MAJOR.MINOR.9999-SNAPSHOT build to Clojars so the very latest version of HoneySQL is always available either via that [snapshot on Clojars](https://clojars.org/com.github.seancorfield/honeysql) or via a git dependency on the latest SHA.
|
||||
|
||||
HoneySQL 2.x requires Clojure 1.9 or later.
|
||||
HoneySQL 2.7.y requires Clojure 1.10.3 or later.
|
||||
Earlier versions of HoneySQL support Clojure 1.9.0.
|
||||
It also supports recent versions of ClojureScript and Babashka.
|
||||
|
||||
Compared to 1.x, HoneySQL 2.x provides a streamlined codebase and a simpler method for extending the DSL. It also supports SQL dialects out-of-the-box and will be extended to support vendor-specific language features over time (unlike 1.x).
|
||||
Compared to the [legacy 1.x version](#1.x), HoneySQL 2.x provides a streamlined codebase and a simpler method for extending the DSL. It also supports SQL dialects out-of-the-box and will be extended to support vendor-specific language features over time (unlike 1.x).
|
||||
|
||||
> Note: you can use 1.x and 2.x side-by-side as they use different group IDs and different namespaces. This allows for a piecemeal migration. See this [summary of differences between 1.x and 2.x](doc/differences-from-1-x.md) if you are migrating from 1.x!
|
||||
|
||||
|
|
@ -31,12 +37,6 @@ Sample code in this documentation is verified via
|
|||
|
||||
Some of these samples show pretty-printed SQL: HoneySQL 2.x supports `:pretty true` which inserts newlines between clauses in the generated SQL strings.
|
||||
|
||||
### HoneySQL 1.x
|
||||
|
||||
[](https://clojars.org/honeysql/honeysql) [](https://cljdoc.org/d/honeysql/honeysql/CURRENT)
|
||||
|
||||
HoneySQL 1.x will continue to get critical security fixes but otherwise should be considered "legacy" at this point.
|
||||
|
||||
## Usage
|
||||
|
||||
This section includes a number of usage examples but does not dive deep into the
|
||||
|
|
@ -48,11 +48,11 @@ section of the documentation before trying to use HoneySQL to build your own que
|
|||
From Clojure:
|
||||
<!-- {:test-doc-blocks/reader-cond :clj} -->
|
||||
```clojure
|
||||
(refer-clojure :exclude '[filter for group-by into partition-by set update])
|
||||
(refer-clojure :exclude '[assert distinct filter for group-by into partition-by set update])
|
||||
(require '[honey.sql :as sql]
|
||||
;; CAUTION: this overwrites several clojure.core fns:
|
||||
;;
|
||||
;; filter, for, group-by, into, partition-by, set, and update
|
||||
;; distinct, filter, for, group-by, into, partition-by, set, and update
|
||||
;;
|
||||
;; you should generally only refer in the specific
|
||||
;; helpers that you want to use!
|
||||
|
|
@ -139,6 +139,24 @@ Namespace-qualified keywords (and symbols) are generally treated as table-qualif
|
|||
=> ["SELECT foo.a, foo.b, foo.c FROM foo WHERE foo.a = ?" "baz"]
|
||||
```
|
||||
|
||||
As of 2.6.1126, there is a helper macro you can use with quoted symbolic
|
||||
queries (that are purely literal, not programmatically constructed) to
|
||||
provide "escape hatches" for certain symbols that you want to be treated
|
||||
as locally bound symbols (and, hence, their values):
|
||||
|
||||
<!-- :test-doc-blocks/skip -->
|
||||
```clojure
|
||||
;; quoted symbolic query with local substitution:
|
||||
(let [search-value "baz"]
|
||||
(sql/formatv [search-value]
|
||||
'{select (foo/a, foo/b, foo/c)
|
||||
from (foo)
|
||||
where (= foo/a search-value)}))
|
||||
=> ["SELECT foo.a, foo.b, foo.c FROM foo WHERE foo.a = ?" "baz"]
|
||||
```
|
||||
|
||||
> Note: this is a Clojure-only feature and is not available in ClojureScript, and it is intended for literal, inline symbolic queries only, not for programmatically constructed queries (where you would be able to substitute the values directly, as you build the query).
|
||||
|
||||
Documentation for the entire data DSL can be found in the
|
||||
[Clause Reference](doc/clause-reference.md), the
|
||||
[Operator Reference](doc/operator-reference.md), and the
|
||||
|
|
@ -199,6 +217,24 @@ If you want to replace a clause, you can `dissoc` the existing clause first, sin
|
|||
=> ["SELECT * FROM foo WHERE (a = ?) AND (b < ?)" 1 100]
|
||||
```
|
||||
|
||||
The power of this approach comes from the abiliity to programmatically and
|
||||
conditionally build up queries:
|
||||
|
||||
<!-- :test-doc-blocks/skip -->
|
||||
```clojure
|
||||
(defn fetch-user [& {:keys [id name]}]
|
||||
(-> (select :*)
|
||||
(from :users)
|
||||
(cond->
|
||||
id (where [:= :id id])
|
||||
name (where [:= :name name]))
|
||||
sql/format))
|
||||
```
|
||||
|
||||
You can call `fetch-user` with either `:id` or `:name` _or both_ and get back
|
||||
a query with the appropriate `WHERE` clause, since the helpers will merge the
|
||||
conditions into the query DSL.
|
||||
|
||||
Column and table names may be aliased by using a vector pair of the original
|
||||
name and the desired alias:
|
||||
|
||||
|
|
@ -210,6 +246,21 @@ name and the desired alias:
|
|||
=> ["SELECT a, b AS bar, c, d AS x FROM foo AS quux WHERE (quux.a = ?) AND (bar < ?)" 1 100]
|
||||
```
|
||||
|
||||
or conditionally:
|
||||
|
||||
<!-- :test-doc-blocks/skip -->
|
||||
```clojure
|
||||
(-> (select :a [:b :bar])
|
||||
(cond->
|
||||
need-c (select :c)
|
||||
x-val (select [:d :x]))
|
||||
(from [:foo :quux])
|
||||
(where [:= :quux.a 1] [:< :bar 100])
|
||||
(cond->
|
||||
x-val (where [:> :x x-val]))
|
||||
sql/format)
|
||||
```
|
||||
|
||||
In particular, note that `(select [:a :b])` means `SELECT a AS b` rather than
|
||||
`SELECT a, b` -- helpers like `select` are generally variadic and do not take
|
||||
a collection of column names.
|
||||
|
|
@ -237,8 +288,7 @@ then provide a collection of rows, each a collection of column values:
|
|||
["Jane" "Daniels" 56]])
|
||||
(sql/format {:pretty true}))
|
||||
=> ["
|
||||
INSERT INTO properties
|
||||
(name, surname, age)
|
||||
INSERT INTO properties (name, surname, age)
|
||||
VALUES (?, ?, ?), (?, ?, ?), (?, ?, ?)
|
||||
"
|
||||
"Jon" "Smith" 34 "Andrew" "Cooper" 12 "Jane" "Daniels" 56]
|
||||
|
|
@ -250,8 +300,7 @@ VALUES (?, ?, ?), (?, ?, ?), (?, ?, ?)
|
|||
["Jane" "Daniels" 56]]}
|
||||
(sql/format {:pretty true}))
|
||||
=> ["
|
||||
INSERT INTO properties
|
||||
(name, surname, age)
|
||||
INSERT INTO properties (name, surname, age)
|
||||
VALUES (?, ?, ?), (?, ?, ?), (?, ?, ?)
|
||||
"
|
||||
"Jon" "Smith" 34 "Andrew" "Cooper" 12 "Jane" "Daniels" 56]
|
||||
|
|
@ -268,8 +317,8 @@ Alternately, you can simply specify the values as maps:
|
|||
{:name "Jane" :surname "Daniels" :age 56}])
|
||||
(sql/format {:pretty true}))
|
||||
=> ["
|
||||
INSERT INTO properties
|
||||
(name, surname, age) VALUES (?, ?, ?), (?, ?, ?), (?, ?, ?)
|
||||
INSERT INTO properties (name, surname, age)
|
||||
VALUES (?, ?, ?), (?, ?, ?), (?, ?, ?)
|
||||
"
|
||||
"John" "Smith" 34
|
||||
"Andrew" "Cooper" 12
|
||||
|
|
@ -281,8 +330,8 @@ INSERT INTO properties
|
|||
{:name "Jane", :surname "Daniels", :age 56}]}
|
||||
(sql/format {:pretty true}))
|
||||
=> ["
|
||||
INSERT INTO properties
|
||||
(name, surname, age) VALUES (?, ?, ?), (?, ?, ?), (?, ?, ?)
|
||||
INSERT INTO properties (name, surname, age)
|
||||
VALUES (?, ?, ?), (?, ?, ?), (?, ?, ?)
|
||||
"
|
||||
"John" "Smith" 34
|
||||
"Andrew" "Cooper" 12
|
||||
|
|
@ -302,8 +351,8 @@ a set of column names that should get the value `DEFAULT` instead of `NULL`:
|
|||
{:name "Jane" :surname "Daniels"}])
|
||||
(sql/format {:pretty true}))
|
||||
=> ["
|
||||
INSERT INTO properties
|
||||
(name, surname, age) VALUES (?, ?, ?), (?, NULL, ?), (?, ?, NULL)
|
||||
INSERT INTO properties (name, surname, age)
|
||||
VALUES (?, ?, ?), (?, NULL, ?), (?, ?, NULL)
|
||||
"
|
||||
"John" "Smith" 34
|
||||
"Andrew" 12
|
||||
|
|
@ -314,8 +363,8 @@ INSERT INTO properties
|
|||
{:name "Jane" :surname "Daniels"}])
|
||||
(sql/format {:pretty true :values-default-columns #{:age}}))
|
||||
=> ["
|
||||
INSERT INTO properties
|
||||
(name, surname, age) VALUES (?, ?, ?), (?, NULL, ?), (?, ?, DEFAULT)
|
||||
INSERT INTO properties (name, surname, age)
|
||||
VALUES (?, ?, ?), (?, NULL, ?), (?, ?, DEFAULT)
|
||||
"
|
||||
"John" "Smith" 34
|
||||
"Andrew" 12
|
||||
|
|
@ -337,8 +386,8 @@ The column values do not have to be literals, they can be nested queries:
|
|||
(sql/format {:pretty true})))
|
||||
|
||||
=> ["
|
||||
INSERT INTO user_profile_to_role
|
||||
(user_profile_id, role_id) VALUES (?, (SELECT id FROM role WHERE name = ?))
|
||||
INSERT INTO user_profile_to_role (user_profile_id, role_id)
|
||||
VALUES (?, (SELECT id FROM role WHERE name = ?))
|
||||
"
|
||||
12345
|
||||
"user"]
|
||||
|
|
@ -352,8 +401,8 @@ INSERT INTO user_profile_to_role
|
|||
:where [:= :name "user"]}}]}
|
||||
(sql/format {:pretty true})))
|
||||
=> ["
|
||||
INSERT INTO user_profile_to_role
|
||||
(user_profile_id, role_id) VALUES (?, (SELECT id FROM role WHERE name = ?))
|
||||
INSERT INTO user_profile_to_role (user_profile_id, role_id)
|
||||
VALUES (?, (SELECT id FROM role WHERE name = ?))
|
||||
"
|
||||
12345
|
||||
"user"]
|
||||
|
|
@ -394,8 +443,7 @@ Composite types are supported:
|
|||
["large" (composite 10 "feet")]])
|
||||
(sql/format {:pretty true}))
|
||||
=> ["
|
||||
INSERT INTO comp_table
|
||||
(name, comp_column)
|
||||
INSERT INTO comp_table (name, comp_column)
|
||||
VALUES (?, (?, ?)), (?, (?, ?))
|
||||
"
|
||||
"small" 1 "inch" "large" 10 "feet"]
|
||||
|
|
@ -407,8 +455,7 @@ VALUES (?, (?, ?)), (?, (?, ?))
|
|||
["large" (composite 10 "feet")]])
|
||||
(sql/format {:pretty true :numbered true}))
|
||||
=> ["
|
||||
INSERT INTO comp_table
|
||||
(name, comp_column)
|
||||
INSERT INTO comp_table (name, comp_column)
|
||||
VALUES ($1, ($2, $3)), ($4, ($5, $6))
|
||||
"
|
||||
"small" 1 "inch" "large" 10 "feet"]
|
||||
|
|
@ -419,8 +466,7 @@ VALUES ($1, ($2, $3)), ($4, ($5, $6))
|
|||
["large" [:composite 10 "feet"]]]}
|
||||
(sql/format {:pretty true}))
|
||||
=> ["
|
||||
INSERT INTO comp_table
|
||||
(name, comp_column)
|
||||
INSERT INTO comp_table (name, comp_column)
|
||||
VALUES (?, (?, ?)), (?, (?, ?))
|
||||
"
|
||||
"small" 1 "inch" "large" 10 "feet"]
|
||||
|
|
@ -517,11 +563,11 @@ If you want to delete everything from a table, you can use `truncate`:
|
|||
```clojure
|
||||
(-> (truncate :films)
|
||||
(sql/format))
|
||||
=> ["TRUNCATE films"]
|
||||
=> ["TRUNCATE TABLE films"]
|
||||
;; or as pure data DSL:
|
||||
(-> {:truncate :films}
|
||||
(sql/format))
|
||||
=> ["TRUNCATE films"]
|
||||
=> ["TRUNCATE TABLE films"]
|
||||
```
|
||||
|
||||
### Set operations
|
||||
|
|
@ -566,6 +612,11 @@ keywords that begin with `%` are interpreted as SQL function calls:
|
|||
=> ["SELECT COUNT(*) FROM foo"]
|
||||
```
|
||||
```clojure
|
||||
;; with an alias:
|
||||
(-> (select [:%count.* :total]) (from :foo) sql/format)
|
||||
=> ["SELECT COUNT(*) AS total FROM foo"]
|
||||
```
|
||||
```clojure
|
||||
(-> (select :%max.id) (from :foo) sql/format)
|
||||
=> ["SELECT MAX(id) FROM foo"]
|
||||
```
|
||||
|
|
@ -579,6 +630,10 @@ regular function calls in a select:
|
|||
=> ["SELECT COUNT(*) FROM foo"]
|
||||
```
|
||||
```clojure
|
||||
(-> (select [[:count :*] :total]) (from :foo) sql/format)
|
||||
=> ["SELECT COUNT(*) AS total FROM foo"]
|
||||
```
|
||||
```clojure
|
||||
(-> (select [:%count.*]) (from :foo) sql/format)
|
||||
=> ["SELECT COUNT(*) FROM foo"]
|
||||
;; or even:
|
||||
|
|
@ -588,20 +643,40 @@ regular function calls in a select:
|
|||
```clojure
|
||||
(-> (select [[:max :id]]) (from :foo) sql/format)
|
||||
=> ["SELECT MAX(id) FROM foo"]
|
||||
(-> (select [[:max :id] :highest]) (from :foo) sql/format)
|
||||
=> ["SELECT MAX(id) AS highest FROM foo"]
|
||||
;; the pure data DSL requires an extra level of brackets:
|
||||
(-> {:select [[[:max :id]]], :from [:foo]} sql/format)
|
||||
=> ["SELECT MAX(id) FROM foo"]
|
||||
(-> {:select [[[:max :id] :highest]], :from [:foo]} sql/format)
|
||||
=> ["SELECT MAX(id) AS highest FROM foo"]
|
||||
;; the shorthand makes this simpler:
|
||||
(-> {:select [[:%max.id]], :from [:foo]} sql/format)
|
||||
=> ["SELECT MAX(id) FROM foo"]
|
||||
;; or even:
|
||||
(-> {:select [[:%max.id :highest]], :from [:foo]} sql/format)
|
||||
=> ["SELECT MAX(id) AS highest FROM foo"]
|
||||
;; or even (no alias):
|
||||
(-> {:select [:%max.id], :from [:foo]} sql/format)
|
||||
=> ["SELECT MAX(id) FROM foo"]
|
||||
;; or even:
|
||||
;; or even (no alias, no other columns):
|
||||
(-> {:select :%max.id, :from :foo} sql/format)
|
||||
=> ["SELECT MAX(id) FROM foo"]
|
||||
```
|
||||
|
||||
Custom columns using functions are built with the same vector format.
|
||||
Be sure to properly nest the vectors so that the first element in the selection
|
||||
is the custom function and the second is the column alias.
|
||||
```clojure
|
||||
(sql/format
|
||||
{:select [:job_name ;; A bare field selection
|
||||
[[:avg [:/ [:- :end_time :start_time] 1000.0]] ;; A custom function
|
||||
:avg_exec_time_seconds ;; The column alias
|
||||
]]
|
||||
:from [:job_data]
|
||||
:group-by :job_name})
|
||||
=> ["SELECT job_name, AVG((end_time - start_time) / ?) AS avg_exec_time_seconds FROM job_data GROUP BY job_name" 1000.0]
|
||||
```
|
||||
|
||||
If a keyword begins with `'`, the function name is formatted as a SQL
|
||||
entity rather than being converted to uppercase and having hyphens `-`
|
||||
converted to spaces). That means that hyphens `-` will become underscores `_`
|
||||
|
|
@ -733,8 +808,8 @@ have a lot of function calls needed in code:
|
|||
[:cast 4325 :integer]]}])
|
||||
(sql/format {:pretty true}))
|
||||
=> ["
|
||||
INSERT INTO sample
|
||||
(location) VALUES (ST_SETSRID(ST_MAKEPOINT(?, ?), CAST(? AS INTEGER)))
|
||||
INSERT INTO sample (location)
|
||||
VALUES (ST_SETSRID(ST_MAKEPOINT(?, ?), CAST(? AS INTEGER)))
|
||||
"
|
||||
0.291 32.621 4325]
|
||||
```
|
||||
|
|
@ -746,7 +821,7 @@ be quoted according to the selected dialect. If you override the dialect in a
|
|||
`format` call, by passing the `:dialect` option, SQL entity names will be automatically
|
||||
quoted. You can override the dialect and turn off quoting by passing `:quoted false`.
|
||||
Valid `:dialect` options are `:ansi` (the default, use this for PostgreSQL),
|
||||
`:mysql`, `:oracle`, or `:sqlserver`:
|
||||
`:mysql`, `:oracle`, or `:sqlserver`. As of 2.5.1091, `:nrql` is also supported:
|
||||
|
||||
```clojure
|
||||
(-> (select :foo.a)
|
||||
|
|
@ -755,6 +830,15 @@ Valid `:dialect` options are `:ansi` (the default, use this for PostgreSQL),
|
|||
(sql/format {:dialect :mysql}))
|
||||
=> ["SELECT `foo`.`a` FROM `foo` WHERE `foo`.`a` = ?" "baz"]
|
||||
```
|
||||
```clojure
|
||||
(-> (select :foo.a)
|
||||
(from :foo)
|
||||
(where [:= :foo.a "baz"])
|
||||
(sql/format {:dialect :nrql}))
|
||||
=> ["SELECT `foo.a` FROM foo WHERE `foo.a` = 'baz'"]
|
||||
```
|
||||
|
||||
See [New Relic NRQL Support](nrsql.md) for more details of the NRQL dialect.
|
||||
|
||||
#### Locking
|
||||
|
||||
|
|
@ -960,8 +1044,15 @@ You can also register SQL clauses, specifying the keyword, the formatting functi
|
|||
|
||||
If you find yourself registering an operator, a function (syntax), or a new clause, consider submitting a [pull request to HoneySQL](https://github.com/seancorfield/honeysql/pulls) so others can use it, too. If it is dialect-specific, let me know in the pull request.
|
||||
|
||||
<a name="1.x"/>
|
||||
## HoneySQL 1.x (legacy)
|
||||
|
||||
[](https://clojars.org/honeysql/honeysql) [](https://cljdoc.org/d/honeysql/honeysql/CURRENT)
|
||||
|
||||
HoneySQL 1.x will continue to get critical security fixes but otherwise should be considered "legacy" at this point.
|
||||
|
||||
## License
|
||||
|
||||
Copyright (c) 2020-2022 Sean Corfield. HoneySQL 1.x was copyright (c) 2012-2020 Justin Kramer and Sean Corfield.
|
||||
Copyright (c) 2020-2024 Sean Corfield. HoneySQL 1.x was copyright (c) 2012-2020 Justin Kramer and Sean Corfield.
|
||||
|
||||
Distributed under the Eclipse Public License, the same as Clojure.
|
||||
|
|
|
|||
8
bb.edn
Normal file
8
bb.edn
Normal file
|
|
@ -0,0 +1,8 @@
|
|||
{:paths ["src"]
|
||||
:tasks
|
||||
{test
|
||||
{:extra-paths ["test"]
|
||||
:extra-deps {io.github.cognitect-labs/test-runner
|
||||
{:git/tag "v0.5.1" :git/sha "dfb30dd"}}
|
||||
:task (exec 'cognitect.test-runner.api/test)
|
||||
:exec-args {:patterns ["^(?!honey.cache).*-test$"]}}}}
|
||||
56
build.clj
56
build.clj
|
|
@ -7,7 +7,7 @@
|
|||
|
||||
Run tests:
|
||||
clojure -X:test
|
||||
clojure -X:test:master
|
||||
clojure -X:test:1.12
|
||||
|
||||
For more information, run:
|
||||
|
||||
|
|
@ -19,7 +19,7 @@
|
|||
[deps-deploy.deps-deploy :as dd]))
|
||||
|
||||
(def lib 'com.github.seancorfield/honeysql)
|
||||
(defn- the-version [patch] (format "2.4.%s" patch))
|
||||
(defn- the-version [patch] (format "2.7.%s" patch))
|
||||
(def version (the-version (b/git-count-revs nil)))
|
||||
(def snapshot (the-version "9999-SNAPSHOT"))
|
||||
(def class-dir "target/classes")
|
||||
|
|
@ -29,8 +29,7 @@
|
|||
(let [basis (b/create-basis {:aliases aliases})
|
||||
combined (t/combine-aliases basis aliases)
|
||||
cmds (b/java-command
|
||||
{:basis basis
|
||||
:java-opts (:jvm-opts combined)
|
||||
{:basis basis
|
||||
:main 'clojure.main
|
||||
:main-args (:main-opts combined)})
|
||||
{:keys [exit]} (b/process cmds)]
|
||||
|
|
@ -48,14 +47,13 @@
|
|||
"Generate and run doc tests.
|
||||
|
||||
Optionally specify :aliases vector:
|
||||
[:1.9] -- test against Clojure 1.9 (the default)
|
||||
[:1.10] -- test against Clojure 1.10.3
|
||||
[:1.10] -- test against Clojure 1.10.3 (the default)
|
||||
[:1.11] -- test against Clojure 1.11.0
|
||||
[:master] -- test against Clojure 1.12 master snapshot
|
||||
[:1.12] -- test against Clojure 1.12.0
|
||||
[:cljs] -- test against ClojureScript"
|
||||
[{:keys [aliases] :as opts}]
|
||||
(gen-doc-tests opts)
|
||||
(run-task (-> [:test :test-doc]
|
||||
(run-task (-> [:test :runner :test-doc]
|
||||
(into aliases)
|
||||
(into (if (some #{:cljs} aliases)
|
||||
[:test-doc-cljs]
|
||||
|
|
@ -63,42 +61,60 @@
|
|||
opts)
|
||||
|
||||
(defn test "Run basic tests." [opts]
|
||||
(run-task [:test :1.11])
|
||||
(run-task [:test :runner :1.11])
|
||||
(run-task [:test :runner :cljs])
|
||||
opts)
|
||||
|
||||
(defn- pom-template [version]
|
||||
[[:description "SQL as Clojure data structures."]
|
||||
[:url "https://github.com/seancorfield/honeysql"]
|
||||
[:licenses
|
||||
[:license
|
||||
[:name "Eclipse Public License"]
|
||||
[:url "http://www.eclipse.org/legal/epl-v10.html"]]]
|
||||
[:developers
|
||||
[:developer
|
||||
[:name "Sean Corfield"]]
|
||||
[:developer
|
||||
[:name "Justin Kramer"]]]
|
||||
[:scm
|
||||
[:url "https://github.com/seancorfield/honeysql"]
|
||||
[:connection "scm:git:https://github.com/seancorfield/honeysql.git"]
|
||||
[:developerConnection "scm:git:ssh:git@github.com:seancorfield/honeysql.git"]
|
||||
[:tag (str "v" version)]]])
|
||||
|
||||
(defn- jar-opts [opts]
|
||||
(let [version (if (:snapshot opts) snapshot version)]
|
||||
(println "\nVersion:" version)
|
||||
(assoc opts
|
||||
:lib lib :version version
|
||||
:jar-file (format "target/%s-%s.jar" lib version)
|
||||
:scm {:tag (str "v" version)}
|
||||
:basis (b/create-basis {})
|
||||
:lib lib :version version
|
||||
:jar-file (format "target/%s-%s.jar" lib version)
|
||||
:basis (b/create-basis {})
|
||||
:class-dir class-dir
|
||||
:target "target"
|
||||
:src-dirs ["src"]
|
||||
:src-pom "template/pom.xml")))
|
||||
:target "target"
|
||||
:src-dirs ["src"]
|
||||
:pom-data (pom-template version))))
|
||||
|
||||
(defn ci
|
||||
"Run the CI pipeline of tests (and build the JAR).
|
||||
|
||||
Default Clojure version is 1.9.0 (:1.9) so :elide
|
||||
Default Clojure version is 1.10.3 (:1.10) so :elide
|
||||
tests for #409 on that version."
|
||||
[opts]
|
||||
(let [aliases [:cljs :elide :1.10 :1.11 :master]
|
||||
(let [aliases [:cljs :elide :1.11 :1.12]
|
||||
opts (jar-opts opts)]
|
||||
(b/delete {:path "target"})
|
||||
(doseq [alias aliases]
|
||||
(run-doc-tests {:aliases [alias]}))
|
||||
(eastwood opts)
|
||||
(doseq [alias aliases]
|
||||
(run-task [:test alias]))
|
||||
(run-task [:test :runner alias]))
|
||||
(b/delete {:path "target"})
|
||||
(println "\nWriting pom.xml...")
|
||||
(b/write-pom opts)
|
||||
(println "\nCopying source...")
|
||||
(b/copy-dir {:src-dirs ["src"] :target-dir class-dir})
|
||||
(println "\nBuilding JAR...")
|
||||
(println "\nBuilding" (:jar-file opts) "...")
|
||||
(b/jar opts))
|
||||
opts)
|
||||
|
||||
|
|
|
|||
|
|
@ -7,14 +7,17 @@
|
|||
success-marker (fs/file target "SUCCESS")
|
||||
docs ["README.md"
|
||||
"doc/clause-reference.md"
|
||||
"doc/databases.md"
|
||||
"doc/differences-from-1-x.md"
|
||||
"doc/extending-honeysql.md"
|
||||
"doc/general-reference.md"
|
||||
"doc/getting-started.md"
|
||||
"doc/nrql.md"
|
||||
;;"doc/operator-reference.md"
|
||||
"doc/options.md"
|
||||
"doc/postgresql.md"
|
||||
"doc/special-syntax.md"]
|
||||
"doc/special-syntax.md"
|
||||
"doc/xtdb.md"]
|
||||
regen-reason (if (not (fs/exists? success-marker))
|
||||
"a previous successful gen result not found"
|
||||
(let [newer-thans (fs/modified-since target
|
||||
|
|
|
|||
23
deps.edn
23
deps.edn
|
|
@ -1,18 +1,16 @@
|
|||
{:mvn/repos {"sonatype" {:url "https://oss.sonatype.org/content/repositories/snapshots/"}}
|
||||
:paths ["src"]
|
||||
:deps {org.clojure/clojure {:mvn/version "1.9.0"}}
|
||||
:deps {org.clojure/clojure {:mvn/version "1.10.3"}}
|
||||
:aliases
|
||||
{;; for help: clojure -A:deps -T:build help/doc
|
||||
:build {:deps {io.github.clojure/tools.build
|
||||
{:git/tag "v0.9.2" :git/sha "fe6b140"}
|
||||
slipset/deps-deploy {:mvn/version "0.2.0"}}
|
||||
:build {:deps {io.github.clojure/tools.build {:mvn/version "0.10.8"}
|
||||
slipset/deps-deploy {:mvn/version "0.2.2"}}
|
||||
:ns-default build}
|
||||
|
||||
;; versions to test against:
|
||||
:1.9 {:override-deps {org.clojure/clojure {:mvn/version "1.9.0"}}}
|
||||
:1.10 {:override-deps {org.clojure/clojure {:mvn/version "1.10.3"}}}
|
||||
:1.11 {:override-deps {org.clojure/clojure {:mvn/version "1.11.1"}}}
|
||||
:master {:override-deps {org.clojure/clojure {:mvn/version "1.12.0-master-SNAPSHOT"}}}
|
||||
:1.11 {:override-deps {org.clojure/clojure {:mvn/version "1.11.4"}}}
|
||||
:1.12 {:override-deps {org.clojure/clojure {:mvn/version "1.12.0"}}}
|
||||
|
||||
:elide ; to test #409 (assertion on helper docstrings)
|
||||
{:jvm-opts ["-Dclojure.compiler.elide-meta=[:doc]"]}
|
||||
|
|
@ -23,16 +21,17 @@
|
|||
:extra-deps {io.github.cognitect-labs/test-runner
|
||||
{:git/tag "v0.5.1" :git/sha "dfb30dd"}
|
||||
org.clojure/core.cache {:mvn/version "RELEASE"}}
|
||||
:main-opts ["-m" "cognitect.test-runner"]
|
||||
:exec-fn cognitect.test-runner.api/test}
|
||||
:runner
|
||||
{:main-opts ["-m" "cognitect.test-runner"]}
|
||||
|
||||
;; various "runners" for tests/CI:
|
||||
:cljs {:extra-deps {olical/cljs-test-runner {:mvn/version "3.8.0"}}
|
||||
:cljs {:extra-deps {olical/cljs-test-runner {:mvn/version "3.8.1"}}
|
||||
:main-opts ["-m" "cljs-test-runner.main"]}
|
||||
|
||||
:gen-doc-tests {:replace-paths ["build"]
|
||||
:extra-deps {babashka/fs {:mvn/version "0.2.12"}
|
||||
com.github.lread/test-doc-blocks {:mvn/version "1.0.166-alpha"}}
|
||||
:extra-deps {babashka/fs {:mvn/version "0.5.24"}
|
||||
com.github.lread/test-doc-blocks {:mvn/version "1.1.20"}}
|
||||
:main-opts ["-m" "honey.gen-doc-tests"]}
|
||||
|
||||
:test-doc {:replace-paths ["src" "target/test-doc-blocks/test"]}
|
||||
|
|
@ -42,5 +41,5 @@
|
|||
"-c" "{:warnings,{:single-segment-namespace,false}}"
|
||||
"-d" "target/test-doc-blocks/test"]}
|
||||
|
||||
:eastwood {:extra-deps {jonase/eastwood {:mvn/version "1.3.0"}}
|
||||
:eastwood {:extra-deps {jonase/eastwood {:mvn/version "1.4.3"}}
|
||||
:main-opts ["-m" "eastwood.lint" "{:source-paths,[\"src\"]}"]}}}
|
||||
|
|
|
|||
|
|
@ -98,7 +98,7 @@ names: the "from" and the "to" names.
|
|||
|
||||
> Note: `:modify-column` is MySQL-specific and should be considered legacy and deprecated. `:alter-column` will produce `MODIFY COLUMN` when the MySQL dialect is selected.
|
||||
|
||||
### add-index, drop-index
|
||||
### add-index, drop-index, create-index
|
||||
|
||||
Used with `:alter-table`,
|
||||
`:add-index` accepts a single (function) expression
|
||||
|
|
@ -125,6 +125,38 @@ user=> (-> (h/alter-table :fruit)
|
|||
["ALTER TABLE fruit ADD PRIMARY KEY(id)"]
|
||||
```
|
||||
|
||||
Some databases treat the standalone `:create-index` differently (e.g. PostgreSQL) while some treat it as an alias to `:alter-table` `:add-index` (e.g. MySQL). It accepts a pair of index specification and column specification:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:create-index [:my-idx [:fruit :appearance]]})
|
||||
["CREATE INDEX my_idx ON fruit (appearance)"]
|
||||
user=> (sql/format {:create-index [[:unique :another-idx] [:fruit :color :appearance]]})
|
||||
["CREATE UNIQUE INDEX another_idx ON fruit (color, appearance)"]
|
||||
```
|
||||
|
||||
PostgreSQL supports IF NOT EXISTS and expressions instead of columns. This may make `:create-index` more useful than `:add-index`:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format (h/create-index [:unique :another-idx :if-not-exists] [:fruit :color :%lower.appearance]))
|
||||
["CREATE UNIQUE INDEX IF NOT EXISTS another_idx ON fruit (color, LOWER(appearance))"]
|
||||
```
|
||||
|
||||
As of 2.6.1147, `USING GIN` index creation is also possible using the keyword
|
||||
`:using-gin` after the table name (or the symbol `using-gin`):
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:create-index [:my-idx [:fruit :using-gin :appearance]]})
|
||||
["CREATE INDEX my_idx ON fruit USING GIN (appearance)"]
|
||||
```
|
||||
|
||||
As of 2.7.next, `USING HASH` index creation is also possible using the keyword
|
||||
`:using-hash` after the table name (or the symbol `using-hash`):
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:create-index [:my-idx [:fruit :using-hash :appearance]]})
|
||||
["CREATE INDEX my_idx ON fruit USING HASH (appearance)"]
|
||||
```
|
||||
|
||||
### rename-table
|
||||
|
||||
Used with `:alter-table`,
|
||||
|
|
@ -229,7 +261,8 @@ CREATE TABLE quux
|
|||
`:create-table-as` can accept a single table name or a sequence
|
||||
that starts with a table name, optionally followed by
|
||||
a flag indicating the creation should be conditional
|
||||
(`:if-not-exists` or the symbol `if-not-exists`),
|
||||
(`:if-not-exists` or the symbol `if-not-exists` or,
|
||||
for BigQuery `:or-replace` or the symbol `or-replace`),
|
||||
optionally followed by a `{:columns ..}` clause to specify
|
||||
the columns to use in the created table, optionally followed
|
||||
by special syntax to specify `TABLESPACE` etc.
|
||||
|
|
@ -261,7 +294,7 @@ A more concise version of the above can use the `TABLE` clause:
|
|||
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:create-table-as [:metro :if-not-exists
|
||||
user=> (sql/format {:create-table-as [:metro :or-replace
|
||||
{:columns [:foo :bar :baz]}
|
||||
[:tablespace [:entity :quux]]],
|
||||
:table :cities,
|
||||
|
|
@ -269,7 +302,7 @@ user=> (sql/format {:create-table-as [:metro :if-not-exists
|
|||
:with-data false}
|
||||
{:pretty true})
|
||||
["
|
||||
CREATE TABLE IF NOT EXISTS metro (foo, bar, baz) TABLESPACE quux AS
|
||||
CREATE OR REPLACE TABLE metro (foo, bar, baz) TABLESPACE quux AS
|
||||
TABLE cities
|
||||
WHERE metroflag = ?
|
||||
WITH NO DATA
|
||||
|
|
@ -284,7 +317,13 @@ will be turned into SQL keywords (this is true for all of the
|
|||
{:create-table-as [:temp :metro :if-not-exists [..]] ..}
|
||||
```
|
||||
|
||||
to produce `CREATE TEMP TABLE IF NOT EXISTS metro ..`.
|
||||
to produce `CREATE TEMP TABLE IF NOT EXISTS metro ..`, or:
|
||||
|
||||
```
|
||||
{:create-table-as [:temp :metro :or-replace [..]] ..}
|
||||
```
|
||||
|
||||
to produce `CREATE OR REPLACE TEMP TABLE metro ..`.
|
||||
|
||||
## create-extension
|
||||
|
||||
|
|
@ -318,6 +357,18 @@ user=> (sql/format {:refresh-materialized-view [:concurrently :products]
|
|||
["REFRESH MATERIALIZED VIEW CONCURRENTLY products WITH NO DATA"]
|
||||
```
|
||||
|
||||
PostgreSQL does not support `IF NOT EXISTS` on `CREATE VIEW` (it supports it on
|
||||
`CREATE MATERIALIZED VIEW`!) so, as of 2.4.1066, HoneySQL also has
|
||||
`:create-or-replace-view` for this case:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:create-or-replace-view [:products]
|
||||
:select [:*]
|
||||
:from [:items]
|
||||
:where [:= :category "product"]})
|
||||
["CREATE OR REPLACE VIEW products AS SELECT * FROM items WHERE category = ?" "product"]
|
||||
```
|
||||
|
||||
## drop-table, drop-extension, drop-view, drop-materialized-view
|
||||
|
||||
`:drop-table` et al can accept a single table (extension, view) name or a sequence of
|
||||
|
|
@ -431,6 +482,39 @@ user=> (sql/format {:with [[[:stuff {:columns [:id :name]}]
|
|||
["WITH stuff (id, name) AS (VALUES (?, ?), (?, ?)) SELECT id, name FROM stuff" 1 "Sean" 2 "Jay"]
|
||||
```
|
||||
|
||||
> Note: you must use the vector-of-vectors format for `:values` here -- if you try to use the vector-of-maps format, `VALUES` will be preceded by the column names (keys from the maps) and the resultant SQL will be invalid.
|
||||
|
||||
You can specify `MATERIALIZED`, `NOT MATERIALIZED` for the CTE:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:with [[:stuff {:select :*
|
||||
:from :table} :not-materialized]]
|
||||
:select :*
|
||||
:from :stuff})
|
||||
["WITH stuff AS NOT MATERIALIZED (SELECT * FROM table) SELECT * FROM stuff"]
|
||||
```
|
||||
|
||||
As of 2.6.1203, you can specify `SEARCH` and/or `CYCLE` clauses, in place of
|
||||
or following the `MATERIALIZED` marker:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:with-recursive [[:stuff {:select :*
|
||||
:from :table}
|
||||
:search-depth-first-by :col :set :search-col]]
|
||||
:select :*
|
||||
:from :stuff})
|
||||
["WITH RECURSIVE stuff AS (SELECT * FROM table) SEARCH DEPTH FIRST BY col SET search_col SELECT * FROM stuff"]
|
||||
```
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:with-recursive [[:stuff {:select :*
|
||||
:from :table}
|
||||
:cycle [:a :b :c] :set :d :to [:abs :e] :default 42 :using :x]]
|
||||
:select :*
|
||||
:from :stuff})
|
||||
["WITH RECURSIVE stuff AS (SELECT * FROM table) CYCLE a, b, c SET d TO ABS(e) DEFAULT ? USING x SELECT * FROM stuff" 42]
|
||||
```
|
||||
|
||||
`:with-recursive` follows the same rules as `:with` and produces `WITH RECURSIVE` instead of just `WITH`.
|
||||
|
||||
## intersect, union, union-all, except, except-all
|
||||
|
|
@ -447,6 +531,13 @@ user=> (sql/format '{union [{select (id,status) from (table-a)}
|
|||
|
||||
> Note: different databases have different precedence rules for these set operations when used in combination -- you may need to use `:nest` to add `(` .. `)` in order to combine these operations in a single SQL statement, if the natural order produced by HoneySQL does not work "as expected" for your database.
|
||||
|
||||
```clojure
|
||||
;; BigQuery requires UNION clauses be parenthesized:
|
||||
user=> (sql/format '{union [{:nest {select (id,status) from (table-a)}}
|
||||
{:nest {select (id,(event status) from (table-b))}}]})
|
||||
["(SELECT id, status FROM table_a) UNION (SELECT id, event AS status, from, table_b)"]
|
||||
```
|
||||
|
||||
## select, select-distinct, table
|
||||
|
||||
`:select` and `:select-distinct` expect a sequence of SQL entities (column names
|
||||
|
|
@ -476,8 +567,40 @@ Here, `:select` has a three expressions as its argument. The first is
|
|||
a simple column name. The second is an expression and its alias. The
|
||||
third is a simple column name and its alias.
|
||||
|
||||
An alias can be a simple name (a keyword or a symbol) or a string. An alias
|
||||
containing a dot (`.`) is treated as a single name for quoting purposes.
|
||||
Otherwise, a simple name will be formatted using table and column name rules
|
||||
(including `-` to `_` translation). An alias specified as a string will not get
|
||||
the `-` to `_` translation. There may be other contexts where you need to
|
||||
refer to an alias but don't want the table/column rules applied to it, e.g.,
|
||||
in an `:order-by` clause. You can use the special syntax `[:alias :some.thing]`
|
||||
to tell HoneySQL to treat `:some.thing` as an alias instead of a table/column
|
||||
name reference.
|
||||
|
||||
`:select-distinct` works the same way but produces `SELECT DISTINCT`.
|
||||
|
||||
As of 2.5.1091, you can use metadata on the argument to `:select` to
|
||||
provide qualifiers for the `SELECT` clause:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select ^:distinct [:id :name] :from :table})
|
||||
["SELECT DISTINCT id, name FROM table"]
|
||||
```
|
||||
|
||||
The metadata can also be a map, with `true` values ignored (which is why
|
||||
`^:distinct` produces just `DISTINCT` even though it is short for
|
||||
`^{:distinct true}`):
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select ^{:as :struct} [:id :name] :from :table})
|
||||
["SELECT AS STRUCT id, name FROM table"]
|
||||
```
|
||||
|
||||
As of 2.5.1103, HoneySQL ignores the following metadata: `:file`, `:line`,
|
||||
`:column`, `:end-line`, and `:end-column` (2.5.1091 only ignored `:line`
|
||||
and `:column`). You can ask HoneySQL to ignore other metadata by specifying
|
||||
the `:ignored-metadata` option to `honey.sql/format`.
|
||||
|
||||
> Google BigQuery support: to provide `SELECT * EXCEPT ..` and `SELECT * REPLACE ..` syntax, HoneySQL supports a vector starting with `:*` or the symbol `*` followed by except columns and/or replace expressions as columns:
|
||||
|
||||
```clojure
|
||||
|
|
@ -492,6 +615,15 @@ user=> (sql/format {:select [[:* :except [:a :b] :replace [[[:inline 2] :c]]]] :
|
|||
The `:table` clause is equivalent to `:select :* :from` and accepts just
|
||||
a simple table name -- see `:create-table-as` above for an example.
|
||||
|
||||
Some databases support inheritance and you can `SELECT .. FROM ONLY ..` or
|
||||
`.. JOIN ONLY ..` to restrict the query to just the specified table. You can
|
||||
use function syntax for this `[:only table]` will produce `ONLY(table)`. This
|
||||
is the ANSI SQL syntax (but PostgreSQL allows the parentheses to be omitted,
|
||||
if you are writing SQL by hand).
|
||||
|
||||
Some databases support temporal queries -- see the `:for` clause section
|
||||
of the `FROM` clause below.
|
||||
|
||||
## select-distinct-on
|
||||
|
||||
Similar to `:select-distinct` above but the first element
|
||||
|
|
@ -546,13 +678,14 @@ user=> (sql/format '{select * bulk-collect-into [arrv 100] from mytable})
|
|||
["SELECT * BULK COLLECT INTO arrv LIMIT ? FROM mytable" 100]
|
||||
```
|
||||
|
||||
## insert-into, replace-into
|
||||
## insert-into, replace-into, patch-into
|
||||
|
||||
There are three use cases with `:insert-into`.
|
||||
There are three use cases with `:insert-into` etc.
|
||||
|
||||
The first case takes just a table specifier (either a
|
||||
table name or a table/alias pair),
|
||||
and then you can optionally specify the columns (via a `:columns` clause).
|
||||
and then you can optionally specify the columns (via a `:columns` clause,
|
||||
or via a `:values` clause using hash maps).
|
||||
|
||||
The second case takes a pair of a table specifier (either a
|
||||
table name or table/alias pair) and a sequence of column
|
||||
|
|
@ -562,7 +695,12 @@ The third case takes a pair of either a table specifier
|
|||
or a table/column specifier and a SQL query.
|
||||
|
||||
For the first and second cases, you'll use the `:values` clause
|
||||
to specify rows of values to insert.
|
||||
to specify rows of values to insert. See [**values**](#values) below
|
||||
for more detail on the `:values` clause.
|
||||
|
||||
`:patch-into` is only supported by XTDB but is
|
||||
part of HoneySQL's "core" dialect anyway. It produces a `PATCH INTO`
|
||||
statement but otherwise has identical syntax to `:insert-into`.
|
||||
|
||||
`:replace-into` is only supported by MySQL and SQLite but is
|
||||
part of HoneySQL's "core" dialect anyway. It produces a `REPLACE INTO`
|
||||
|
|
@ -607,7 +745,33 @@ user=> (sql/format '{insert-into (((transport t) (id, name)) {select (*) from (c
|
|||
["INSERT INTO transport AS t (id, name) SELECT * FROM cars"]
|
||||
```
|
||||
|
||||
> Note: if you specify `:columns` for an `:insert-into` that also includes column names, you will get invalid SQL. Similarly, if you specify `:columns` when `:values` is based on hash maps, you will get invalid SQL. Since clauses are generated independently, there is no cross-checking performed if you provide an illegal combination of clauses.
|
||||
Some databases do not let you override (insert) values that would override
|
||||
generated column values, unless your SQL specifies `OVERRIDING SYSTEM VALUE`
|
||||
or `OVERRIDING USER VALUE`. As of 2.4.1066, you can use `:overriding-value` as
|
||||
an option to `:insert-into` to specify this, with either `:system` or `:user`
|
||||
as the option's value. The options can be specified as a hash map in the
|
||||
first position of the `:insert-into` clause, prior to the table specifier.
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:insert-into [{:overriding-value :system}
|
||||
[:transport :t] [:id :name]]
|
||||
:values [[1 "Car"] [2 "Boat"] [3 "Bike"]]}
|
||||
{:pretty true})
|
||||
["
|
||||
INSERT INTO transport AS t (id, name) OVERRIDING SYSTEM VALUE
|
||||
VALUES (?, ?), (?, ?), (?, ?)
|
||||
" 1 "Car" 2 "Boat" 3 "Bike"]
|
||||
user=> (sql/format {:insert-into [{:overriding-value :user}
|
||||
[:transport :t] [:id :name]]
|
||||
:values [[1 "Car"] [2 "Boat"] [3 "Bike"]]}
|
||||
{:pretty true})
|
||||
["
|
||||
INSERT INTO transport AS t (id, name) OVERRIDING USER VALUE
|
||||
VALUES (?, ?), (?, ?), (?, ?)
|
||||
" 1 "Car" 2 "Boat" 3 "Bike"]
|
||||
```
|
||||
|
||||
> Note: as of 2.4.1066, if you specify `:columns` for an `:insert-into` that also includes column names, or with a `:values` clause based on hash maps (which imply column names), then an order of precedence is applied: the columns specified directly in `:insert-into` take precedence, then the `:columns` clause, then the implied column names from the `:values` clause. Prior to 2.4.1066, you would get invalid SQL generated.
|
||||
|
||||
## update
|
||||
|
||||
|
|
@ -621,7 +785,28 @@ user=> (sql/format {:update :transport
|
|||
["UPDATE transport SET name = ? WHERE id = ?" "Yacht" 2]
|
||||
```
|
||||
|
||||
## delete, delete-from
|
||||
You can also set columns to `NULL` or to their default values:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:update :transport
|
||||
:set {:owner nil, :date_built [:default]}
|
||||
:where [:= :id 2]})
|
||||
["UPDATE transport SET owner = NULL, date_built = DEFAULT WHERE id = ?"
|
||||
2]
|
||||
```
|
||||
|
||||
You can also `UPDATE .. FROM (VALUES ..) ..` where you might also need `:composite`:
|
||||
|
||||
```clojure
|
||||
(sql/format {:update :table :set {:a :v.a}
|
||||
:from [[{:values [[1 2 3]
|
||||
[4 5 6]]}
|
||||
[:v [:composite :a :b :c]]]]
|
||||
:where [:and [:= :x :v.b] [:> :y :v.c]]})
|
||||
["UPDATE table SET a = v.a FROM (VALUES (?, ?, ?), (?, ?, ?)) AS v (a, b, c) WHERE (x = v.b) AND (y > v.c)" 1 2 3 4 5 6]
|
||||
```
|
||||
|
||||
## delete, delete-from, erase-from
|
||||
|
||||
`:delete-from` is the simple use case here, accepting just a
|
||||
SQL entity (table name). `:delete` allows for deleting from
|
||||
|
|
@ -638,16 +823,30 @@ user=> (sql/format {:delete [:order :item]
|
|||
["DELETE order, item FROM order INNER JOIN item ON order.item_id = item.id WHERE item.id = ?" 42]
|
||||
```
|
||||
|
||||
`:erase-from` is only supported by XTDB and produces an `ERASE FROM`
|
||||
statement but otherwise has identical syntax to `:delete-from`. It
|
||||
is a "hard" delete as opposed to a temporal delete.
|
||||
|
||||
## truncate
|
||||
|
||||
`:truncate` accepts a simple SQL entity (table name)
|
||||
or a table name followed by various options:
|
||||
or a table name followed by various options, or a
|
||||
sequence that starts with a sequence of one or more table names,
|
||||
optionally followed by various options:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format '{truncate transport})
|
||||
["TRUNCATE transport"]
|
||||
["TRUNCATE TABLE transport"]
|
||||
user=> (sql/format '{truncate (transport)})
|
||||
["TRUNCATE TABLE transport"]
|
||||
user=> (sql/format '{truncate (transport restart identity)})
|
||||
["TRUNCATE transport RESTART IDENTITY"]
|
||||
["TRUNCATE TABLE transport RESTART IDENTITY"]
|
||||
user=> (sql/format '{truncate ((transport))})
|
||||
["TRUNCATE TABLE transport"]
|
||||
user=> (sql/format '{truncate ((transport other))})
|
||||
["TRUNCATE TABLE transport, other"]
|
||||
user=> (sql/format '{truncate ((transport other) restart identity)})
|
||||
["TRUNCATE TABLE transport, other RESTART IDENTITY"]
|
||||
```
|
||||
|
||||
## columns
|
||||
|
|
@ -675,8 +874,9 @@ user=> (sql/format {:update :order
|
|||
|
||||
`:from` accepts a single sequence argument that lists
|
||||
one or more SQL entities. Each entity can either be a
|
||||
simple table name (keyword or symbol) or a pair of a
|
||||
table name and an alias:
|
||||
simple table name (keyword or symbol) or a sequence of a
|
||||
table name, followed by an optional alias, followed by an
|
||||
optional temporal clause:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select [:username :name]
|
||||
|
|
@ -691,8 +891,59 @@ user=> (sql/format {:select [:u.username :s.name]
|
|||
["SELECT u.username, s.name FROM user AS u, status AS s WHERE (u.statusid = s.id) AND (u.id = ?)" 9]
|
||||
```
|
||||
|
||||
`:from` can also accept a `:values` clause:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:update :table :set {:a :v.a}
|
||||
:from [[{:values [[1 2 3]
|
||||
[4 5 6]]}
|
||||
[:v [:composite :a :b :c]]]]
|
||||
:where [:and [:= :x :v.b] [:> :y :v.c]]})
|
||||
["UPDATE table SET a = v.a FROM (VALUES (?, ?, ?), (?, ?, ?)) AS v (a, b, c) WHERE (x = v.b) AND (y > v.c)" 1 2 3 4 5 6]
|
||||
```
|
||||
|
||||
As of 2.4.1066, HoneySQL supports a temporal clause that starts with `:for`,
|
||||
followed by the time reference
|
||||
(e.g., `:system-time` or `:business-time`), followed by a temporal qualifier,
|
||||
one of:
|
||||
* `:all`
|
||||
* `:as-of timestamp`
|
||||
* `:from timestamp1 :to timestamp2`
|
||||
* `:between timestamp1 :and timestamp2`
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select [:username]
|
||||
:from [[:user :for :system-time :as-of [:inline "2019-08-01 15:23:00"]]]
|
||||
:where [:= :id 9]})
|
||||
["SELECT username FROM user FOR SYSTEM_TIME AS OF '2019-08-01 15:23:00' WHERE id = ?" 9]
|
||||
user=> (sql/format {:select [:u.username]
|
||||
:from [[:user :u :for :system-time :from [:inline "2019-08-01 15:23:00"] :to [:inline "2019-08-01 15:24:00"]]]
|
||||
:where [:= :u.id 9]})
|
||||
["SELECT u.username FROM user FOR SYSTEM_TIME FROM '2019-08-01 15:23:00' TO '2019-08-01 15:24:00' AS u WHERE u.id = ?" 9]
|
||||
```
|
||||
|
||||
As of 2.6.1126, HoneySQL supports metadata on a table expression to provide
|
||||
database-specific hints, such as SQL Server's `WITH (..)` clause:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select [:col]
|
||||
:from [^:nolock [:table]]
|
||||
:where [:= :id 9]})
|
||||
["SELECT col FROM table WITH (NOLOCK) WHERE id = ?" 9]
|
||||
user=> (sql/format {:select [:col]
|
||||
:from [^:nolock [:table :t]]
|
||||
:where [:= :id 9]})
|
||||
["SELECT col FROM table AS t WITH (NOLOCK) WHERE id = ?" 9]
|
||||
```
|
||||
|
||||
Since you cannot put metadata on a keyword, the table name must be written as
|
||||
a vector even when you have no alias.
|
||||
|
||||
> Note: the actual formatting of a `:from` clause is currently identical to the formatting of a `:select` clause.
|
||||
|
||||
If you are using inheritance, you can specify `ONLY(table)` as a function
|
||||
call: `[:only :table]`.
|
||||
|
||||
## using
|
||||
|
||||
`:using` accepts a single sequence argument that lists
|
||||
|
|
@ -700,7 +951,7 @@ one or more SQL entities. Each entity can either be a
|
|||
simple table name (keyword or symbol) or a pair of a
|
||||
table name and an alias.
|
||||
|
||||
`:using` is intended to be used as a simple join with a `:delete-from`
|
||||
`:using` is intended to be used as a simple join, for example with a `:delete-from`
|
||||
clause (see [PostgreSQL DELETE statement](https://www.postgresql.org/docs/12/sql-delete.html)
|
||||
for more detail).
|
||||
|
||||
|
|
@ -778,6 +1029,31 @@ user=> (sql/format {:select [:t.ref :pp.code]
|
|||
["SELECT t.ref, pp.code FROM transaction AS t LEFT JOIN paypal_tx AS pp USING (id) WHERE ? = pp.status" "settled"]
|
||||
```
|
||||
|
||||
As of 2.6.1126, HoneySQL supports metadata on a table expression to provide
|
||||
database-specific hints, such as SQL Server's `WITH (..)` clause:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select [:col]
|
||||
:from [:table]
|
||||
:join [^:nolock [:extra] [:= :table.extra_id :extra.id]]
|
||||
:where [:= :id 9]})
|
||||
["SELECT col FROM table INNER JOIN extra WITH (NOLOCK) ON table.extra_id = extra.id WHERE id = ?" 9]
|
||||
user=> (sql/format {:select [:col]
|
||||
:from [[:table :t]]
|
||||
:join [^:nolock [:extra :x] [:= :t.extra_id :x.id]]
|
||||
:where [:= :id 9]})
|
||||
["SELECT col FROM table AS t INNER JOIN extra AS x WITH (NOLOCK) ON t.extra_id = x.id WHERE id = ?" 9]
|
||||
```
|
||||
|
||||
Since you cannot put metadata on a keyword, the table name must be written as
|
||||
a vector even when you have no alias.
|
||||
|
||||
If you are using inheritance, you can specify `ONLY(table)` as a function
|
||||
call: `[:only :table]`.
|
||||
|
||||
See also the [`:join` special syntax](https://cljdoc.org/d/com.github.seancorfield/honeysql/CURRENT/doc/getting-started/sql-special-syntax-#join)
|
||||
for nested `JOIN` expressions.
|
||||
|
||||
## cross-join
|
||||
|
||||
`:cross-join` accepts a single sequence argument that lists
|
||||
|
|
@ -812,6 +1088,9 @@ The `:where` clause can have a single SQL expression, or
|
|||
a sequence of SQL expressions prefixed by either `:and`
|
||||
or `:or`. See examples of `:where` in various clauses above.
|
||||
|
||||
If `:where` is given an empty sequence, the `WHERE` clause will
|
||||
be omitted from the generated SQL.
|
||||
|
||||
Sometimes it is convenient to construct a `WHERE` clause that
|
||||
tests several columns for equality, and you might have a Clojure
|
||||
hash map containing those values. `honey.sql/map=` exists to
|
||||
|
|
@ -833,6 +1112,11 @@ user=> (sql/format '{select (*) from (table)
|
|||
["SELECT * FROM table GROUP BY status, YEAR(created_date)"]
|
||||
```
|
||||
|
||||
You can `GROUP BY` expressions, column names (`:col1`), or table and column (`:table.col1`),
|
||||
or aliases (`:some.alias`). Since there is ambiguity between the formatting
|
||||
of those, you can use the special syntax `[:alias :some.thing]` to tell
|
||||
HoneySQL to treat `:some.thing` as an alias instead of a table/column name.
|
||||
|
||||
## having
|
||||
|
||||
The `:having` clause works identically to `:where` above
|
||||
|
|
@ -840,7 +1124,7 @@ but is rendered into the SQL later in precedence order.
|
|||
|
||||
## window, partition-by (and over)
|
||||
|
||||
`:window` accepts a pair of SQL entity (the window name)
|
||||
`:window` accept alternating pairs of SQL entity (the window name)
|
||||
and the window "function" as a SQL clause (a hash map).
|
||||
|
||||
`:partition-by` accepts the same arguments as `:select` above
|
||||
|
|
@ -866,6 +1150,25 @@ SELECT id, AVG(salary) OVER (PARTITION BY department ORDER BY designation ASC) A
|
|||
FROM employee
|
||||
WINDOW w AS (PARTITION BY department)
|
||||
"]
|
||||
;; multiple windows:
|
||||
user=> (sql/format {:select [:id
|
||||
[[:over
|
||||
[[:avg :salary]
|
||||
{:partition-by [:department]
|
||||
:order-by [:designation]}
|
||||
:Average]
|
||||
[[:max :salary]
|
||||
:w
|
||||
:MaxSalary]]]]
|
||||
:from [:employee]
|
||||
:window [:w {:partition-by [:department]}
|
||||
:x {:partition-by [:salary]}]}
|
||||
{:pretty true})
|
||||
["
|
||||
SELECT id, AVG(salary) OVER (PARTITION BY department ORDER BY designation ASC) AS Average, MAX(salary) OVER w AS MaxSalary
|
||||
FROM employee
|
||||
WINDOW w AS (PARTITION BY department), x AS (PARTITION BY salary)
|
||||
"]
|
||||
;; easier to write with helpers (and easier to read!):
|
||||
user=> (sql/format (-> (select :id
|
||||
(over [[:avg :salary] (-> (partition-by :department) (order-by :designation)) :Average]
|
||||
|
|
@ -878,6 +1181,18 @@ SELECT id, AVG(salary) OVER (PARTITION BY department ORDER BY designation ASC) A
|
|||
FROM employee
|
||||
WINDOW w AS (PARTITION BY department)
|
||||
"]
|
||||
;; multiple window clauses:
|
||||
user=> (sql/format (-> (select :id
|
||||
(over [[:avg :salary] (-> (partition-by :department) (order-by :designation)) :Average]
|
||||
[[:max :salary] :w :MaxSalary]))
|
||||
(from :employee)
|
||||
(window :w (partition-by :department))
|
||||
(window :x (partition-by :salary))) {:pretty true})
|
||||
["
|
||||
SELECT id, AVG(salary) OVER (PARTITION BY department ORDER BY designation ASC) AS Average, MAX(salary) OVER w AS MaxSalary
|
||||
FROM employee
|
||||
WINDOW w AS (PARTITION BY department), x AS (PARTITION BY salary)
|
||||
"]
|
||||
```
|
||||
|
||||
The window function in the `:over` expression may be `{}` or `nil`:
|
||||
|
|
@ -901,14 +1216,30 @@ user=> (sql/format (-> (select :id
|
|||
["SELECT id, AVG(salary) OVER () AS Average, MAX(salary) OVER () AS MaxSalary FROM employee"]
|
||||
```
|
||||
|
||||
## distinct, expr
|
||||
|
||||
Related to the windowing clauses above, `:distinct` and `:expr` are
|
||||
intended to let you mix clauses with expressions, such as in BigQuery's
|
||||
`ARRAY_AGG` function:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select [[[:over
|
||||
[[:array_agg {:distinct [:ignore-nulls :col] :order-by :x}]
|
||||
{:partition-by :y}]]]]})
|
||||
["SELECT ARRAY_AGG (DISTINCT col IGNORE NULLS ORDER BY x ASC) OVER (PARTITION BY y)"]
|
||||
```
|
||||
|
||||
## order-by
|
||||
|
||||
`:order-by` accepts a sequence of one or more ordering
|
||||
`:order-by` accepts a sequence of zero or more ordering
|
||||
expressions. Each ordering expression is either a simple
|
||||
SQL entity or a pair of a SQL expression and a direction
|
||||
(which can be `:asc`, `:desc`, `:nulls-first`, `:desc-null-last`,
|
||||
etc -- or the symbol equivalent).
|
||||
|
||||
If `:order-by` is given an empty sequence, the `ORDER BY` clause will
|
||||
be omitted from the generated SQL.
|
||||
|
||||
If you want to order by an expression, you should wrap it
|
||||
as a pair with a direction:
|
||||
|
||||
|
|
@ -939,6 +1270,11 @@ user=> (sql/format {:select [:*] :from :table
|
|||
["SELECT * FROM table ORDER BY CASE WHEN NOW() < expiry_date THEN created_date ELSE expiry_date END DESC"]
|
||||
```
|
||||
|
||||
You can `ORDER BY` column names (`:col1`), or table and column (`:table.col1`),
|
||||
or aliases (`:some.alias`). Since there is ambiguity between the formatting
|
||||
of those, you can use the special syntax `[:alias :some.thing]` to tell
|
||||
HoneySQL to treat `:some.thing` as an alias instead of a table/column name.
|
||||
|
||||
## limit, offset, fetch
|
||||
|
||||
Some databases, including MySQL, support `:limit` and `:offset`
|
||||
|
|
@ -1022,16 +1358,59 @@ same as the `:for` clause above.
|
|||
row values or a sequence of sequences, also representing row
|
||||
values.
|
||||
|
||||
In the former case, all of the rows are augmented to have
|
||||
### values with hash maps
|
||||
|
||||
If you provide a sequence of hash maps, the `:values` clause
|
||||
will generate a `VALUES` clause, and will also generate the column names
|
||||
as part of the `INSERT INTO` (or `REPLACE INTO`) statement.
|
||||
|
||||
If there is no `INSERT INTO` (or `REPLACE INTO`) statement in the context
|
||||
of the `:values` clause, the column names will be generated as a part of
|
||||
the `VALUES` clause itself.
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:values [{:col-a 1 :col-b 2}]})
|
||||
["(col_a, col_b) VALUES (?, ?)" 1 2]
|
||||
```
|
||||
|
||||
In addition, all of the rows are augmented to have
|
||||
either `NULL` or `DEFAULT` values for any missing keys (columns).
|
||||
By default, `NULL` is used but you can specify a set of columns
|
||||
to get `DEFAULT` values, via the `:values-default-columns` option.
|
||||
You can also be explicit and use `[:default]` as a value to generate `DEFAULT`.
|
||||
In the latter case -- a sequence of sequences --
|
||||
all of the rows are padded to the same length by adding `nil`
|
||||
|
||||
### values with sequences
|
||||
|
||||
If you provide a sequence of sequences, the `:values` clause
|
||||
will generate a `VALUES` clause with no column names and the
|
||||
row values following.
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:values [[1 2]]})
|
||||
["VALUES (?, ?)" 1 2]
|
||||
```
|
||||
|
||||
In addition, all of the rows are padded to the same length by adding `nil`
|
||||
values if needed (since `:values` does not know how or if column
|
||||
names are being used in this case).
|
||||
|
||||
### values row (MySQL)
|
||||
|
||||
MySQL supports `VALUES` as a table expression in multiple
|
||||
contexts, and it uses "row constructors" to represent the
|
||||
rows of values.
|
||||
|
||||
HoneySQL supports this by using the keyword `:row` (or
|
||||
symbol `'row`) as the first element of a sequence of values.
|
||||
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:values [:row [1 2] [3 4]]})
|
||||
["VALUES ROW(?, ?), ROW(?, ?)" 1 2 3 4]
|
||||
```
|
||||
|
||||
### values examples
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:insert-into :table
|
||||
:values [[1 2] [2 3 4 5] [3 4 5]]})
|
||||
|
|
@ -1071,7 +1450,7 @@ user=> (sql/format {:insert-into :table
|
|||
:values [{:a 1 :b 2 :c 3}
|
||||
:default
|
||||
{:a 4 :b 5 :c 6}]})
|
||||
["INSERT INTO table (a, b, c) VALUES (?, ?, ?), DEFAULT, (?, ?, ?)" 6 5 4]
|
||||
["INSERT INTO table (a, b, c) VALUES (?, ?, ?), DEFAULT, (?, ?, ?)" 1 2 3 4 5 6]
|
||||
user=> (sql/format {:insert-into :table
|
||||
:values [[1 2 3] :default [4 5 6]]})
|
||||
["INSERT INTO table VALUES (?, ?, ?), DEFAULT, (?, ?, ?)" 1 2 3 4 5 6]
|
||||
|
|
@ -1084,10 +1463,11 @@ as if they are separate clauses but they will appear
|
|||
in pairs: `ON ... DO ...`.
|
||||
|
||||
`:on-conflict` accepts a sequence of zero or more
|
||||
SQL entities (keywords or symbols), optionally
|
||||
SQL expressions, optionally
|
||||
followed by a single SQL clause (hash map). It can also
|
||||
accept either a single SQL entity or a single SQL clause.
|
||||
The SQL entities are column names and the SQL clause can be an
|
||||
The SQL expressions can be just column names or function calls etc,
|
||||
and the SQL clause can be an
|
||||
`:on-constraint` clause or a`:where` clause.
|
||||
|
||||
_[For convenience of use with the `on-conflict` helper, this clause can also accept any of those arguments, wrapped in a sequence; it can also accept an empty sequence, and just produce `ON CONFLICT`, so that it can be combined with other clauses directly]_
|
||||
|
|
|
|||
|
|
@ -7,6 +7,8 @@
|
|||
["SQL Operator Reference" {:file "doc/operator-reference.md"}]
|
||||
["SQL 'Special Syntax'" {:file "doc/special-syntax.md"}]
|
||||
["PostgreSQL Support" {:file "doc/postgresql.md"}]
|
||||
["XTDB Support" {:file "doc/xtdb.md"}]
|
||||
["New Relic NRQL Support" {:file "doc/nrql.md"}]
|
||||
["Other Databases" {:file "doc/databases.md"}]]
|
||||
["All the Options" {:file "doc/options.md"}]
|
||||
["Extending HoneySQL" {:file "doc/extending-honeysql.md"}]
|
||||
|
|
|
|||
|
|
@ -1,12 +1,15 @@
|
|||
# Other Databases
|
||||
|
||||
There is a dedicated section for [PostgreSQL Support](postgres.md).
|
||||
There are dedicated sections for [New Relic Query Language Support](nrql.md),
|
||||
[PostgreSQL Support](postgres.md), and
|
||||
[XTDB Support](xtdb.md).
|
||||
This section provides hints and tips for generating SQL for other
|
||||
databases.
|
||||
|
||||
As a reminder, HoneySQL supports the following dialects out of the box:
|
||||
* `:ansi` -- which is the default and provides broad support for PostgreSQL as well
|
||||
* `:mysql` -- which includes MariaDB and Percona
|
||||
* `:nrql` -- as of 2.5.1091
|
||||
* `:oracle`
|
||||
* `:sqlserver` -- Microsoft SQL Server
|
||||
|
||||
|
|
|
|||
|
|
@ -9,7 +9,7 @@ The DSL itself -- the data structures that both versions convert to SQL and para
|
|||
If you are using Clojure 1.11, you can invoke `format` with a mixture of named arguments and a trailing hash
|
||||
map of additional options, if you wish.
|
||||
|
||||
HoneySQL 1.x supported Clojure 1.7 and later. HoneySQL 2.x requires Clojure 1.9 or later.
|
||||
HoneySQL 1.x supported Clojure 1.7 and later. HoneySQL 2.7.y requires Clojure 1.10.3 or later. Earlier versions of HoneySQL 2.x support Clojure 1.9.0.
|
||||
|
||||
## Group, Artifact, and Namespaces
|
||||
|
||||
|
|
@ -63,7 +63,7 @@ Supported Clojure versions: 1.7 and later.
|
|||
In `deps.edn`:
|
||||
<!-- :test-doc-blocks/skip -->
|
||||
```clojure
|
||||
com.github.seancorfield/honeysql {:mvn/version "2.4.1011"}
|
||||
com.github.seancorfield/honeysql {:mvn/version "2.7.1295"}
|
||||
```
|
||||
|
||||
Required as:
|
||||
|
|
@ -90,7 +90,7 @@ The new namespaces are:
|
|||
* `honey.sql` -- the primary API (just `format` now),
|
||||
* `honey.sql.helpers` -- helper functions to build the DSL.
|
||||
|
||||
Supported Clojure versions: 1.9 and later.
|
||||
Supported Clojure versions: 1.10.3 and later.
|
||||
|
||||
## API Changes
|
||||
|
||||
|
|
@ -133,7 +133,7 @@ The following new syntax has been added:
|
|||
* `:default` -- for `DEFAULT` values (in inserts) and for declaring column defaults in table definitions,
|
||||
* `:escape` -- used to wrap a regular expression so that non-standard escape characters can be provided,
|
||||
* `:inline` -- used as a function to replace the `sql/inline` / `#sql/inline` machinery,
|
||||
* `:interval` -- used as a function to support `INTERVAL <n> <units>`, e.g., `[:interval 30 :days]` for databases that support it (e.g., MySQL),
|
||||
* `:interval` -- used as a function to support `INTERVAL <n> <units>`, e.g., `[:interval 30 :days]` for databases that support it (e.g., MySQL) and, as of 2.4.1026, for `INTERVAL 'n units'`, e.g., `[:interval "24 hours"]` for ANSI/PostgreSQL.
|
||||
* `:lateral` -- used to wrap a statement or expression, to provide a `LATERAL` join,
|
||||
* `:lift` -- used as a function to prevent interpretation of a Clojure data structure as DSL syntax (e.g., when passing a vector or hash map as a parameter value) -- this should mostly be a replacement for `honeysql.format/value`,
|
||||
* `:nest` -- used as a function to add an extra level of nesting (parentheses) around an expression,
|
||||
|
|
|
|||
|
|
@ -10,7 +10,7 @@ register formatters or behavior corresponding to clauses,
|
|||
operators, and functions.
|
||||
|
||||
Built in clauses include: `:select`, `:from`, `:where` and
|
||||
many more. Built in operators include: `:=`, `:+`, `:mod`.
|
||||
many more. Built in operators include: `:=`, `:+`, `:%`.
|
||||
Built in functions (special syntax) include: `:array`, `:case`,
|
||||
`:cast`, `:inline`, `:raw` and many more.
|
||||
|
||||
|
|
@ -50,6 +50,11 @@ The formatter function will be called with:
|
|||
* The clause name (always as a keyword),
|
||||
* The sequence of arguments provided.
|
||||
|
||||
The formatter function should return a vector whose first element is the
|
||||
generated SQL string and whose remaining elements (if any) are the parameters
|
||||
lifted from the DSL (for which the generated SQL string should contain `?`
|
||||
placeholders).
|
||||
|
||||
The third argument to `register-clause!` allows you to
|
||||
insert your new clause formatter so that clauses are
|
||||
formatted in the correct order for your SQL dialect.
|
||||
|
|
@ -57,6 +62,26 @@ For example, `:select` comes before `:from` which comes
|
|||
before `:where`. You can call `clause-order` to see what the
|
||||
current ordering of clauses is.
|
||||
|
||||
<!-- :test-doc-blocks/skip -->
|
||||
```clojure
|
||||
;; the formatter will be passed your new clause and the value associated
|
||||
;; with that clause in the DSL (which is often a sequence but does not
|
||||
;; need to be -- it can be whatever syntax you desire in the DSL):
|
||||
(sql/register-clause! :foobar
|
||||
(fn [clause x]
|
||||
(let [[sql & params]
|
||||
(if (ident? x)
|
||||
(sql/format-expr x)
|
||||
(sql/format-dsl x))]
|
||||
(c/into [(str (sql/sql-kw clause) " " sql)] params)))
|
||||
:from) ; SELECT ... FOOBAR ... FROM ...
|
||||
;; example usage:
|
||||
(sql/format {:select [:a :b] :foobar :baz})
|
||||
=> ["SELECT a, b FOOBAR baz"]
|
||||
(sql/format {:select [:a :b] :foobar {:where [:= :id 1]}})
|
||||
=> ["SELECT a, b FOOBAR WHERE id = ?" 1]
|
||||
```
|
||||
|
||||
> Note: if you call `register-clause!` more than once for the same clause, the last call "wins". This allows you to correct an incorrect clause order insertion by simply calling `register-clause!` again with a different third argument.
|
||||
|
||||
## Defining a Helper Function for a New Clause
|
||||
|
|
@ -182,6 +207,7 @@ _New in HoneySQL 2.3.x_
|
|||
The built-in dialects that HoneySQL supports are:
|
||||
* `:ansi` -- the default, that quotes SQL entity names with double-quotes, like `"this"`
|
||||
* `:mysql` -- quotes SQL entity names with backticks, and changes the precedence of `SET` in `UPDATE`
|
||||
* `:nrql` -- as of 2.5.1091, see [New Relic NRQL Support](nrsql.md) for more details of the NRQL dialect
|
||||
* `:oracle` -- quotes SQL entity names like `:ansi`, and does not use `AS` in aliases
|
||||
* `:sqlserver` -- quotes SQL entity names with brackets, like `[this]`
|
||||
|
||||
|
|
|
|||
|
|
@ -10,14 +10,14 @@ For the Clojure CLI, add the following dependency to your `deps.edn` file:
|
|||
|
||||
<!-- :test-doc-blocks/skip -->
|
||||
```clojure
|
||||
com.github.seancorfield/honeysql {:mvn/version "2.4.1011"}
|
||||
com.github.seancorfield/honeysql {:mvn/version "2.7.1295"}
|
||||
```
|
||||
|
||||
For Leiningen, add the following dependency to your `project.clj` file:
|
||||
|
||||
<!-- :test-doc-blocks/skip -->
|
||||
```clojure
|
||||
[com.github.seancorfield/honeysql "2.4.1011"]
|
||||
[com.github.seancorfield/honeysql "2.7.1295"]
|
||||
```
|
||||
|
||||
HoneySQL produces SQL statements but does not execute them.
|
||||
|
|
@ -26,13 +26,13 @@ To execute SQL statements, you will also need a JDBC wrapper like
|
|||
|
||||
You can also experiment with HoneySQL directly in a browser -- no installation
|
||||
required -- using [John Shaffer](https://github.com/john-shaffer)'s awesome
|
||||
[HoneySQL web app](https://www.john-shaffer.com/honeysql/), written in ClojureScript!
|
||||
[HoneySQL web app](https://john.shaffe.rs/honeysql/), written in ClojureScript!
|
||||
|
||||
## Basic Concepts
|
||||
|
||||
SQL statements are represented as hash maps, with keys that
|
||||
represent clauses in SQL. SQL expressions are generally
|
||||
represented as sequences, where the first element identifies
|
||||
represented as vectors, where the first element identifies
|
||||
the function or operator and the remaining elements are the
|
||||
arguments or operands.
|
||||
|
||||
|
|
@ -54,11 +54,13 @@ or symbols, are treated as positional parameters and replaced
|
|||
by `?` in the SQL string and lifted out into the vector that
|
||||
is returned from `format`.
|
||||
|
||||
Most clauses expect a sequence as their value, containing
|
||||
Most clauses expect a vector as their value, containing
|
||||
either a list of SQL entities or the representation of a SQL
|
||||
expression. Some clauses accept a single SQL entity. A few
|
||||
accept a more specialized form (such as `:set` accepting a
|
||||
hash map of SQL entities and SQL expressions).
|
||||
accept a more specialized form (such as `:set` within an `:update` clause
|
||||
accepting a hash map of SQL entities and SQL expressions).
|
||||
|
||||
> Note: clauses can have a list as their value, but literal vectors and keywords are easier to type without quoting.
|
||||
|
||||
A SQL entity can be a simple keyword (or symbol) or a pair
|
||||
that represents a SQL entity and its alias (where aliases are allowed):
|
||||
|
|
@ -86,6 +88,8 @@ avoid evaluation:
|
|||
;;=> ["SELECT t.id, name AS item FROM table AS t WHERE id = ?" 1]
|
||||
```
|
||||
|
||||
> Note: these quoted forms may be appealing to users familiar with Datalog-family query languages, and they can be easier to type (and read) in some cases since you do not need to add `:` (shift-`;` on most keyboards) to the start of each SQL entity. The quoted forms do not work well in the [HoneySQL web app](https://john.shaffe.rs/honeysql/) so it's better to stick with vectors and keywords when using that.
|
||||
|
||||
If you wish, you can specify SQL entities as namespace-qualified
|
||||
keywords (or symbols) and the namespace portion will treated as
|
||||
the table name, i.e., `:foo/bar` instead of `:foo.bar`:
|
||||
|
|
@ -101,8 +105,8 @@ the table name, i.e., `:foo/bar` instead of `:foo.bar`:
|
|||
## SQL Expressions
|
||||
|
||||
In addition to using hash maps to describe SQL clauses,
|
||||
HoneySQL uses sequences to describe SQL expressions. Any
|
||||
sequence that begins with a keyword (or symbol) is considered
|
||||
HoneySQL uses vectors to describe SQL expressions. Any
|
||||
vector that begins with a keyword (or symbol) is considered
|
||||
to be a kind of function invocation. Certain "functions" are
|
||||
considered to be "special syntax" and have custom rendering.
|
||||
Some "functions" are considered to be operators. In general,
|
||||
|
|
@ -151,22 +155,22 @@ Some examples:
|
|||
[:now] ;=> "NOW()"
|
||||
[:count :*] ;=> "COUNT(*)"
|
||||
[:or [:<> :name nil] [:= :status-id 0]] ;=> "(name IS NOT NULL) OR (status_id = ?)"
|
||||
;; with a parameter of 0 -- the nil value is inlined as NULL
|
||||
;; the nil value is inlined as NULL but 0 is provided as a parameter
|
||||
```
|
||||
|
||||
`:inline` is an example of "special syntax" and it renders its
|
||||
(single) argument as part of the SQL string generated by `format`.
|
||||
arguments as part of the SQL string generated by `format`.
|
||||
|
||||
Another form of special syntax that is treated as function calls
|
||||
is keywords or symbols that begin with `%`. Such keywords (or symbols)
|
||||
is keywords or symbols that begin with `%`. Such keywords (or quoted symbols)
|
||||
are split at `.` and turned into function calls:
|
||||
|
||||
<!-- :test-doc-blocks/skip -->
|
||||
```clojure
|
||||
%now ;=> NOW()
|
||||
%count.* ;=> COUNT(*)
|
||||
%max.foo ;=> MAX(foo)
|
||||
%f.a.b ;=> F(a,b)
|
||||
:%now ;=> NOW()
|
||||
:%count.* ;=> COUNT(*)
|
||||
:%max.foo ;=> MAX(foo)
|
||||
:%f.a.b ;=> F(a,b)
|
||||
```
|
||||
|
||||
If you need to reference a table or alias for a column, you can use
|
||||
|
|
@ -200,6 +204,8 @@ expression requires an extra level of nesting:
|
|||
;;=> ["SELECT x, y AS d, Z(e), Z(f) AS g"]
|
||||
(sql/format {:select [:x [:y :d] [:%z.e] [:%z.f :g]]})
|
||||
;;=> ["SELECT x, y AS d, Z(e), Z(f) AS g"]
|
||||
(sql/format {:select [:x [:y :d] :%z.e [:%z.f :g]]})
|
||||
;;=> ["SELECT x, y AS d, Z(e), Z(f) AS g"]
|
||||
```
|
||||
|
||||
## SQL Parameters
|
||||
|
|
@ -255,7 +261,7 @@ Or with `:numbered true`:
|
|||
|
||||
## Functional Helpers
|
||||
|
||||
In addition to the hash map (and sequences) approach of building
|
||||
In addition to the hash map (and vectors) approach of building
|
||||
SQL queries with raw Clojure data structures, a
|
||||
[namespace full of helper functions](https://cljdoc.org/d/com.github.seancorfield/honeysql/CURRENT/api/honey.sql.helpers)
|
||||
is also available. These functions are generally variadic and threadable:
|
||||
|
|
@ -274,8 +280,11 @@ is also available. These functions are generally variadic and threadable:
|
|||
There is a helper function for every single clause that HoneySQL
|
||||
supports out of the box. In addition, there are helpers for
|
||||
`composite`, `lateral`, `over`, and `upsert` that make it easier to construct those
|
||||
parts of the SQL DSL (examples of `composite` appear in the [README](/README.md),
|
||||
examples of `over` appear in the [Clause Reference](clause-reference.md))
|
||||
parts of the SQL DSL (examples of `composite` appear in the
|
||||
[README](/README.md#composite-types)
|
||||
and in the [General Reference](general-reference.md#tuples-and-composite-values);
|
||||
examples of `over` appear in the
|
||||
[Clause Reference](clause-reference.md#window-partition-by-and-over))
|
||||
|
||||
In general, `(helper :foo expr)` will produce `{:helper [:foo expr]}`
|
||||
(with a few exceptions -- see the docstring of the helper function
|
||||
|
|
@ -341,13 +350,14 @@ The dialects supported by HoneySQL 2.x are:
|
|||
* `:ansi` -- the default, including most PostgreSQL extensions
|
||||
* `:sqlserver` -- Microsoft SQL Server
|
||||
* `:mysql` -- MySQL (and Percona and MariaDB)
|
||||
* `:nrql` -- as of 2.5.1091
|
||||
* `:oracle` -- Oracle
|
||||
|
||||
The most visible difference between dialects is how SQL entities
|
||||
should be quoted (if the `:quoted true` option is provided to `format`).
|
||||
Most databases use `"` for quoting (the `:ansi` and `:oracle` dialects).
|
||||
The `:sqlserver` dialect uses `[`..`]` and the `:mysql` dialect uses
|
||||
```..```. In addition, the `:oracle` dialect disables `AS` in aliases.
|
||||
`` ` ``..`` ` ``. In addition, the `:oracle` dialect disables `AS` in aliases.
|
||||
|
||||
> Note: by default, quoting is **off** which produces cleaner-looking SQL and assumes you control all the symbols/keywords used as table, column, and function names -- the "SQL entities". If you are building any SQL or DDL where the table, column, or function names could be provided by an external source, **you should specify `:quoted true` to ensure all SQL entities are safely quoted**. As of 2.3.928, if you do _not_ specify `:quoted` as an option, HoneySQL will automatically quote any SQL entities that seem unusual, i.e., that contain any characters that are not alphanumeric or underscore. Purely alphanumeric entities will not be quoted (no entities were quoted by default prior to 2.3.928). You can prevent that auto-quoting by explicitly passing `:quoted false` into the `format` call but, from a security point of view, you should think very carefully before you do that: quoting entity names helps protect you from injection attacks! As of 2.4.947, you can change the default setting of `:quoted` from `nil` to `true` (or `false`) via the `set-options!` function.
|
||||
|
||||
|
|
@ -355,9 +365,11 @@ Currently, the only dialect that has substantive differences from
|
|||
the others is `:mysql` for which the `:set` clause
|
||||
has a different precedence than ANSI SQL.
|
||||
|
||||
See [New Relic NRQL Support](nrsql.md) for more details of the NRQL dialect.
|
||||
|
||||
You can change the dialect globally using the `set-dialect!` function,
|
||||
passing in one of the keywords above. You need to call this function
|
||||
before you call `format` for the first time.
|
||||
before you call `format` for the first time. See below for examples.
|
||||
|
||||
You can change the dialect for a single `format` call by
|
||||
specifying the `:dialect` option in that call.
|
||||
|
|
@ -366,7 +378,8 @@ Alphanumeric SQL entities are not quoted by default but if you specify the
|
|||
dialect in a `format` call, they will be quoted. If you don't
|
||||
specify a dialect in the `format` call, you can specify
|
||||
`:quoted true` to have SQL entities quoted. You can also enable quoting
|
||||
globally via the `set-dialect!` function.
|
||||
globally via the `set-dialect!` function. See below for an example
|
||||
with `:quoted true`.
|
||||
|
||||
If you want to use a dialect _and_ use the default quoting strategy (automatically quote any SQL entities that seem unusual), specify a `:dialect` option and set `:quoted nil`:
|
||||
|
||||
|
|
@ -402,7 +415,8 @@ If you want to use a dialect _and_ use the default quoting strategy (automatical
|
|||
```
|
||||
|
||||
Out of the box, as part of the extended ANSI SQL support,
|
||||
HoneySQL supports quite a few [PostgreSQL extensions](postgresql.md).
|
||||
HoneySQL supports quite a few [PostgreSQL extensions](postgresql.md)
|
||||
and [XTDB extensions](xtdb.md).
|
||||
|
||||
> Note: the [nilenso/honeysql-postgres](https://github.com/nilenso/honeysql-postgres) library which provided PostgreSQL support for HoneySQL 1.x does not work with HoneySQL 2.x. However, HoneySQL 2.x includes all of the functionality from that library (up to 0.4.112) out of the box!
|
||||
|
||||
|
|
|
|||
44
doc/nrql.md
Normal file
44
doc/nrql.md
Normal file
|
|
@ -0,0 +1,44 @@
|
|||
# New Relic NRQL Support
|
||||
|
||||
As of 2.5.1091, HoneySQL provides some support for New Relic's NRQL query language.
|
||||
|
||||
At present, the following additional SQL clauses (and their corresponding
|
||||
helper functions) are supported:
|
||||
|
||||
* `:facet` - implemented just like `:select`
|
||||
* `:since` - implemented like `:interval`
|
||||
* `:until` - implemented like `:interval`
|
||||
* `:compare-with` - implemented like `:interval`
|
||||
* `:timeseries` - implemented like `:interval`
|
||||
|
||||
> Note: `:timeseries :auto` is the shortest way to specify a timeseries.
|
||||
|
||||
When you select the `:nrql` dialect, SQL formatting assumes `:inline true`
|
||||
so that the generated SQL string can be used directly in NRQL queries.
|
||||
|
||||
In addition, stropping (quoting) is done using backticks, like MySQL,
|
||||
but entities are not split at `/` or `.` characters, so that:
|
||||
|
||||
```
|
||||
:foo/bar.baz ;;=> `foo/bar.baz`
|
||||
```
|
||||
|
||||
```clojure
|
||||
user=> (require '[honey.sql :as sql])
|
||||
nil
|
||||
```
|
||||
```clojure
|
||||
user=> (sql/format {:select [:mulog/timestamp :mulog/event-name]
|
||||
:from :Log
|
||||
:where [:= :mulog/data.account "foo-account-id"]
|
||||
:since [2 :days :ago]
|
||||
:limit 2000}
|
||||
{:dialect :nrql :pretty true})
|
||||
["
|
||||
SELECT `mulog/timestamp`, `mulog/event-name`
|
||||
FROM Log
|
||||
WHERE `mulog/data.account` = 'foo-account-id'
|
||||
LIMIT 2000
|
||||
SINCE 2 DAYS AGO
|
||||
"]
|
||||
```
|
||||
|
|
@ -32,15 +32,14 @@ can simply evaluate to `nil` instead).
|
|||
;;=> ["...WHERE (id = ?) OR (type = ?)..." 42 "match"]
|
||||
```
|
||||
|
||||
## in
|
||||
## in, not-in
|
||||
|
||||
Predicate for checking an expression is
|
||||
is a member of a specified set of values.
|
||||
Predicates for checking an expression is or is not a member of a specified set of values.
|
||||
|
||||
The two most common forms are:
|
||||
|
||||
* `[:in :col [val1 val2 ...]]` where the `valN` can be arbitrary expressions,
|
||||
* `[:in :col {:select ...}]` where the `SELECT` specifies a single column.
|
||||
* `[:in :col [val1 val2 ...]]` or `[:not-in :col [val1 val2 ...]]` where the `valN` can be arbitrary expressions,
|
||||
* `[:in :col {:select ...}]` or `[:not-in :col {:select ...}]` where the `SELECT` specifies a single column.
|
||||
|
||||
`:col` could be an arbitrary SQL expression (but is most
|
||||
commonly just a column name).
|
||||
|
|
@ -48,15 +47,17 @@ commonly just a column name).
|
|||
The former produces an inline vector expression with the
|
||||
values resolved as regular SQL expressions (i.e., with
|
||||
literal values lifted out as parameters): `col IN [?, ?, ...]`
|
||||
or `col NOT IN [?, ?, ...]`
|
||||
|
||||
The latter produces a sub-select, as expected: `col IN (SELECT ...)`
|
||||
or `col NOT IN (SELECT ...)`
|
||||
|
||||
You can also specify the set of values via a named parameter:
|
||||
|
||||
* `[:in :col :?values]` where `:params {:values [1 2 ...]}` is provided to `format` in the options.
|
||||
* `[:in :col :?values]` or `[:not-in :col :?values]` where `:params {:values [1 2 ...]}` is provided to `format` in the options.
|
||||
|
||||
In this case, the named parameter is expanded directly when
|
||||
`:in` is formatted to obtain the sequence of values (which
|
||||
`:in` (or `:not-in`) is formatted to obtain the sequence of values (which
|
||||
must be _sequential_, not a Clojure set). That means you
|
||||
cannot use this approach and also specify `:cache` -- see
|
||||
[cache in All the Options](options.md#cache) for more details.
|
||||
|
|
@ -67,9 +68,9 @@ of columns, producing `(col1, col2) IN (SELECT ...)`, but
|
|||
you need to specify the columns (or expressions) using the
|
||||
`:composite` special syntax:
|
||||
|
||||
* `[:in [:composite :col1 :col2] ...]`
|
||||
* `[:in [:composite :col1 :col2] ...]` or `[:not-in [:composite :col1 :col2] ...]`
|
||||
|
||||
This produces `(col1, col2) IN ...`
|
||||
This produces `(col1, col2) IN ...` or `(col1, col2) NOT IN ...`
|
||||
|
||||
> Note: This is a change from HoneySQL 1.x which accepted a sequence of column names but required more work for arbitrary expressions.
|
||||
|
||||
|
|
@ -109,7 +110,7 @@ Predicates for `NULL` and Boolean values:
|
|||
;;=> ["...WHERE col IS NOT FALSE..."]
|
||||
```
|
||||
|
||||
## mod, xor, + - * / % | & ^
|
||||
## xor, + - * / % | & ^
|
||||
|
||||
Mathematical and bitwise operators.
|
||||
|
||||
|
|
@ -120,6 +121,17 @@ as an alias for `regexp`.
|
|||
|
||||
`similar-to` and `not-similar-to` are also supported.
|
||||
|
||||
## with ordinality
|
||||
|
||||
The ANSI SQL `WITH ORDINALITY` expression is supported as an infix operator:
|
||||
|
||||
```clojure
|
||||
{...
|
||||
[:with-ordinality [:jsonb_array_elements :j] [:arr :item :index]]
|
||||
...}
|
||||
;;=> ["...JSONB_ARRAY_ELEMENTS(j) WITH ORDINALITY ARR(item, index)..."]
|
||||
```
|
||||
|
||||
## ||
|
||||
|
||||
String concatenation operator.
|
||||
|
|
|
|||
|
|
@ -18,10 +18,12 @@ All options may be omitted. The default behavior of each option is described in
|
|||
* `:cache` -- an atom containing a [clojure.core.cache](https://github.com/clojure/core.cache) cache used to cache generated SQL; the default behavior is to generate SQL on each call to `format`,
|
||||
* `:checking` -- `:none` (default), `:basic`, or `:strict` to control the amount of lint-like checking that HoneySQL performs,
|
||||
* `:dialect` -- a keyword that identifies a dialect to be used for this specific call to `format`; the default is to use what was specified in `set-dialect!` or `:ansi` if no other dialect has been set,
|
||||
* `:ignored-metadata` -- a sequence of metadata keys that should be ignored when formatting (in addition to `:file`, `:line`, `:column`, `:end-line` and `:end-column` which are always ignored); the default is `[]` -- no additional metadata is ignored (since 2.5.1103),
|
||||
* `:inline` -- a Boolean indicating whether or not to inline parameter values, rather than use `?` placeholders and a sequence of parameter values; the default is `false` -- values are not inlined,
|
||||
* `:numbered` -- a Boolean indicating whether to generate numbered placeholders in the generated SQL (`$1`, `$2`, etc) or positional placeholders (`?`); the default is `false` (positional placeholders); this option was added in 2.4.962,
|
||||
* `:params` -- a hash map providing values for named parameters, identified by names (keywords or symbols) that start with `?` in the DSL; the default is that any such named parameters will have `nil` values,
|
||||
* `:quoted` -- a Boolean indicating whether or not to quote (strop) SQL entity names (table and column names); the default is `nil` -- alphanumeric SQL entity names are not quoted but (as of 2.3.928) "unusual" SQL entity names are quoted; a `false` value turns off all quoting,
|
||||
* `:quoted-always` -- an optional regex that matches SQL entity names that should always be quoted (stropped) regardless of the value of `:quoted`; the default is `nil` -- no SQL entity names are always quoted,
|
||||
* `:quoted-snake` -- a Boolean indicating whether or not quoted and string SQL entity names should have `-` replaced by `_`; the default is `false` -- quoted and string SQL entity names are left exactly as-is,
|
||||
* `:values-default-columns` -- a sequence of column names that should have `DEFAULT` values instead of `NULL` values if used in a `VALUES` clause with no associated matching value in the hash maps passed in; the default behavior is for such missing columns to be given `NULL` values.
|
||||
|
||||
|
|
@ -130,7 +132,10 @@ selected dialect.
|
|||
|
||||
If `:quoted false`, SQL entity names that represent tables and columns
|
||||
will not be quoted. If those SQL entity names are reserved words in
|
||||
SQL, the generated SQL will be invalid.
|
||||
SQL, the generated SQL will be invalid. You can use the `:quoted-always`
|
||||
option to specify a regex, to identify SQL entity names that should
|
||||
always be quoted (stropped) regardless of the value of `:quoted`, e.g.,
|
||||
reserved words that you happen to use as table or column names.
|
||||
|
||||
The quoting (stropping) is dialect-dependent:
|
||||
* `:ansi` -- uses double quotes
|
||||
|
|
|
|||
|
|
@ -37,14 +37,27 @@ Clojure users can opt for the shorter `(require '[honey.sql :as sql] '[honey.sql
|
|||
|
||||
## Working with Arrays
|
||||
|
||||
HoneySQL supports `:array` as special syntax to produce `ARRAY[..]` expressions
|
||||
but PostgreSQL also has an "array constructor" for creating arrays from subquery results.
|
||||
HoneySQL supports `:array` as special syntax to produce `ARRAY[..]` expressions:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select [[[:array [1 2 3]] :a]]})
|
||||
["SELECT ARRAY[?, ?, ?] AS a" 1 2 3]
|
||||
```
|
||||
|
||||
PostgreSQL also has an "array constructor" for creating arrays from subquery results.
|
||||
|
||||
```sql
|
||||
SELECT ARRAY(SELECT oid FROM pg_proc WHERE proname LIKE 'bytea%');
|
||||
```
|
||||
|
||||
In order to produce that SQL, you can use HoneySQL's "as-is" function syntax to circumvent
|
||||
As of 2.5.1091, HoneySQL supports this syntax directly:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select [[[:array {:select :oid :from :pg_proc :where [:like :proname [:inline "bytea%"]]}]]]})
|
||||
["SELECT ARRAY(SELECT oid FROM pg_proc WHERE proname LIKE 'bytea%')"]
|
||||
```
|
||||
|
||||
Prior to 2.5.1091, you had to use HoneySQL's "as-is" function syntax to circumvent
|
||||
the special syntax:
|
||||
|
||||
```clojure
|
||||
|
|
@ -52,13 +65,6 @@ user=> (sql/format {:select [[[:'ARRAY {:select :oid :from :pg_proc :where [:lik
|
|||
["SELECT ARRAY (SELECT oid FROM pg_proc WHERE proname LIKE 'bytea%')"]
|
||||
```
|
||||
|
||||
Compare this with the `ARRAY[..]` syntax:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select [[[:array [1 2 3]] :a]]})
|
||||
["SELECT ARRAY[?, ?, ?] AS a" 1 2 3]
|
||||
```
|
||||
|
||||
## Operators with @, #, and ~
|
||||
|
||||
A number of PostgreSQL operators contain `@`, `#`, or `~` which are not legal in a Clojure keyword or symbol (as literal syntax). The namespace `honey.sql.pg-ops` provides convenient symbolic names for these JSON and regex operators, substituting `at` for `@`, `hash` for `#`, and `tilde` for `~`.
|
||||
|
|
@ -102,8 +108,8 @@ user=> (-> (insert-into :distributors)
|
|||
(returning :*)
|
||||
(sql/format {:pretty true}))
|
||||
["
|
||||
INSERT INTO distributors
|
||||
(did, dname) VALUES (?, ?), (?, ?)
|
||||
INSERT INTO distributors (did, dname)
|
||||
VALUES (?, ?), (?, ?)
|
||||
ON CONFLICT (did)
|
||||
DO UPDATE SET dname = EXCLUDED.dname
|
||||
RETURNING *
|
||||
|
|
@ -124,8 +130,8 @@ user=> (-> (insert-into :distributors)
|
|||
(returning :*)
|
||||
(sql/format {:pretty true}))
|
||||
["
|
||||
INSERT INTO distributors
|
||||
(did, dname) VALUES (?, ?), (?, ?)
|
||||
INSERT INTO distributors (did, dname)
|
||||
VALUES (?, ?), (?, ?)
|
||||
ON CONFLICT (did)
|
||||
DO UPDATE SET dname = EXCLUDED.dname
|
||||
RETURNING *
|
||||
|
|
@ -144,8 +150,8 @@ user=> (-> (insert-into :distributors)
|
|||
do-nothing))
|
||||
(sql/format {:pretty true}))
|
||||
["
|
||||
INSERT INTO distributors
|
||||
(did, dname) VALUES (?, ?)
|
||||
INSERT INTO distributors (did, dname)
|
||||
VALUES (?, ?)
|
||||
ON CONFLICT (did)
|
||||
DO NOTHING
|
||||
"
|
||||
|
|
@ -161,8 +167,8 @@ user=> (-> (insert-into :distributors)
|
|||
do-nothing
|
||||
(sql/format {:pretty true}))
|
||||
["
|
||||
INSERT INTO distributors
|
||||
(did, dname) VALUES (?, ?)
|
||||
INSERT INTO distributors (did, dname)
|
||||
VALUES (?, ?)
|
||||
ON CONFLICT (did)
|
||||
DO NOTHING
|
||||
"
|
||||
|
|
@ -180,8 +186,8 @@ user=> (-> (insert-into :distributors)
|
|||
do-nothing
|
||||
(sql/format {:pretty true}))
|
||||
["
|
||||
INSERT INTO distributors
|
||||
(did, dname) VALUES (?, ?)
|
||||
INSERT INTO distributors (did, dname)
|
||||
VALUES (?, ?)
|
||||
ON CONFLICT ON CONSTRAINT distributors_pkey
|
||||
DO NOTHING
|
||||
"
|
||||
|
|
@ -194,8 +200,8 @@ user=> (-> (insert-into :distributors)
|
|||
do-nothing
|
||||
(sql/format {:pretty true}))
|
||||
["
|
||||
INSERT INTO distributors
|
||||
(did, dname) VALUES (?, ?)
|
||||
INSERT INTO distributors (did, dname)
|
||||
VALUES (?, ?)
|
||||
ON CONFLICT
|
||||
ON CONSTRAINT distributors_pkey
|
||||
DO NOTHING
|
||||
|
|
@ -215,8 +221,8 @@ user=> (-> (insert-into :user)
|
|||
(do-update-set :phone :name (where [:= :user.active false]))
|
||||
(sql/format {:pretty true}))
|
||||
["
|
||||
INSERT INTO user
|
||||
(phone, name) VALUES (?, ?)
|
||||
INSERT INTO user (phone, name)
|
||||
VALUES (?, ?)
|
||||
ON CONFLICT (phone) WHERE phone IS NOT NULL
|
||||
DO UPDATE SET phone = EXCLUDED.phone, name = EXCLUDED.name WHERE user.active = FALSE
|
||||
"
|
||||
|
|
@ -231,8 +237,8 @@ user=> (sql/format
|
|||
:where [:= :user.active false]}}
|
||||
{:pretty true})
|
||||
["
|
||||
INSERT INTO user
|
||||
(phone, name) VALUES (?, ?)
|
||||
INSERT INTO user (phone, name)
|
||||
VALUES (?, ?)
|
||||
ON CONFLICT (phone) WHERE phone IS NOT NULL
|
||||
DO UPDATE SET phone = EXCLUDED.phone, name = EXCLUDED.name WHERE user.active = FALSE
|
||||
"
|
||||
|
|
@ -268,8 +274,8 @@ user=> (-> (insert-into :table)
|
|||
(do-update-set {:counter [:+ :table.counter 1]})
|
||||
(sql/format {:pretty true}))
|
||||
["
|
||||
INSERT INTO table
|
||||
(id, counter) VALUES (?, ?)
|
||||
INSERT INTO table (id, counter)
|
||||
VALUES (?, ?)
|
||||
ON CONFLICT (id)
|
||||
DO UPDATE SET counter = table.counter + ?
|
||||
" "id" 1 1]
|
||||
|
|
@ -280,13 +286,31 @@ user=> (-> {:insert-into :table
|
|||
:do-update-set {:counter [:+ :table.counter 1]}}
|
||||
(sql/format {:pretty true}))
|
||||
["
|
||||
INSERT INTO table
|
||||
(id, counter) VALUES (?, ?)
|
||||
INSERT INTO table (id, counter)
|
||||
VALUES (?, ?)
|
||||
ON CONFLICT (id)
|
||||
DO UPDATE SET counter = table.counter + ?
|
||||
" "id" 1 1]
|
||||
```
|
||||
|
||||
You can use `:EXCLUDED.column` in a hash map to produce the
|
||||
same effect as `:column` in a vector:
|
||||
|
||||
```clojure
|
||||
user=> (-> (insert-into :table)
|
||||
(values [{:id "id" :counter 1}])
|
||||
(on-conflict :id)
|
||||
(do-update-set {:name :EXCLUDED.name
|
||||
:counter [:+ :table.counter 1]})
|
||||
(sql/format {:pretty true}))
|
||||
["
|
||||
INSERT INTO table (id, counter)
|
||||
VALUES (?, ?)
|
||||
ON CONFLICT (id)
|
||||
DO UPDATE SET name = EXCLUDED.name, counter = table.counter + ?
|
||||
" "id" 1 1]
|
||||
```
|
||||
|
||||
If you need to combine a `DO UPDATE SET` hash map expression
|
||||
with a `WHERE` clause, you need to explicitly use the `:fields` /
|
||||
`:where` format explained above. Here's how those two examples
|
||||
|
|
@ -300,8 +324,8 @@ user=> (-> (insert-into :table)
|
|||
:where [:> :table.counter 1]})
|
||||
(sql/format {:pretty true}))
|
||||
["
|
||||
INSERT INTO table
|
||||
(id, counter) VALUES (?, ?)
|
||||
INSERT INTO table (id, counter)
|
||||
VALUES (?, ?)
|
||||
ON CONFLICT (id)
|
||||
DO UPDATE SET counter = table.counter + ? WHERE table.counter > ?
|
||||
" "id" 1 1 1]
|
||||
|
|
@ -313,8 +337,8 @@ user=> (-> {:insert-into :table
|
|||
:where [:> :table.counter 1]}}
|
||||
(sql/format {:pretty true}))
|
||||
["
|
||||
INSERT INTO table
|
||||
(id, counter) VALUES (?, ?)
|
||||
INSERT INTO table (id, counter)
|
||||
VALUES (?, ?)
|
||||
ON CONFLICT (id)
|
||||
DO UPDATE SET counter = table.counter + ? WHERE table.counter > ?
|
||||
" "id" 1 1 1]
|
||||
|
|
@ -432,11 +456,6 @@ user=> (-> (alter-table :fruit)
|
|||
(drop-column :skin)
|
||||
sql/format)
|
||||
["ALTER TABLE fruit DROP COLUMN skin"]
|
||||
;; alter table alter column:
|
||||
user=> (-> (alter-table :fruit)
|
||||
(alter-column :name [:varchar 64] [:not nil])
|
||||
sql/format)
|
||||
["ALTER TABLE fruit ALTER COLUMN name VARCHAR(64) NOT NULL"]
|
||||
;; alter table rename column:
|
||||
user=> (-> (alter-table :fruit)
|
||||
(rename-column :cost :price)
|
||||
|
|
@ -449,6 +468,29 @@ user=> (-> (alter-table :fruit)
|
|||
["ALTER TABLE fruit RENAME TO vegetable"]
|
||||
```
|
||||
|
||||
The following does not work for PostgreSQL, but does work for several other databases:
|
||||
|
||||
```clojure
|
||||
;; alter table alter column:
|
||||
user=> (-> (alter-table :fruit)
|
||||
(alter-column :name [:varchar 64] [:not nil])
|
||||
sql/format)
|
||||
["ALTER TABLE fruit ALTER COLUMN name VARCHAR(64) NOT NULL"]
|
||||
```
|
||||
|
||||
For PostgreSQL, you need separate statements:
|
||||
|
||||
```clojure
|
||||
user=> (-> (alter-table :fruit)
|
||||
(alter-column :name :type [:varchar 64])
|
||||
sql/format)
|
||||
["ALTER TABLE fruit ALTER COLUMN name TYPE VARCHAR(64)"]
|
||||
user=> (-> (alter-table :fruit)
|
||||
(alter-column :name :set [:not nil])
|
||||
sql/format)
|
||||
["ALTER TABLE fruit ALTER COLUMN name SET NOT NULL"]
|
||||
```
|
||||
|
||||
The following PostgreSQL-specific DDL statements are supported
|
||||
(with the same syntax as the nilenso library but `sql/format`
|
||||
takes slightly different options):
|
||||
|
|
|
|||
|
|
@ -6,15 +6,51 @@ as special syntactic forms.
|
|||
|
||||
The first group are used for SQL expressions. The second (last group) are used primarily in column definitions (as part of `:with-columns` and `:add-column` / `:alter-column`).
|
||||
|
||||
## array
|
||||
|
||||
Accepts a single argument, which is expected to evaluate to a sequence,
|
||||
with an optional second argument specifying the type of the array,
|
||||
and produces `ARRAY[?, ?, ..]` for the elements of that sequence (as SQL parameters):
|
||||
The examples in this section assume the following:
|
||||
|
||||
```clojure
|
||||
(require '[honey.sql :as sql])
|
||||
```
|
||||
|
||||
## alias
|
||||
|
||||
Accepts a single argument which should be an alias name (from an `AS` clause
|
||||
elsewhere in the overall SQL statement) and uses alias formatting rules rather
|
||||
than table/column formatting rules (different handling of dots and hyphens).
|
||||
This allows you to override HoneySQL's default assumption about entity names
|
||||
and strings.
|
||||
|
||||
```clojure
|
||||
(sql/format {:select [[:column-name "some-alias"]]
|
||||
:from :b
|
||||
:order-by [[[:alias "some-alias"]]]})
|
||||
;;=> ["SELECT column_name AS \"some-alias\" FROM b ORDER BY \"some-alias\" ASC"]
|
||||
(sql/format {:select [[:column-name :'some-alias]]
|
||||
:from :b
|
||||
:order-by [[[:alias :'some-alias]]]})
|
||||
;;=> ["SELECT column_name AS \"some-alias\" FROM b ORDER BY \"some-alias\" ASC"]
|
||||
(sql/format {:select [[:column-name "some-alias"]]
|
||||
:from :b
|
||||
:group-by [[:alias "some-alias"]]})
|
||||
;;=> ["SELECT column_name AS \"some-alias\" FROM b GROUP BY \"some-alias\""]
|
||||
(sql/format {:select [[:column-name "some-alias"]]
|
||||
:from :b
|
||||
:group-by [[:alias :'some-alias]]})
|
||||
;;=> ["SELECT column_name AS \"some-alias\" FROM b GROUP BY \"some-alias\""]
|
||||
```
|
||||
|
||||
## array
|
||||
|
||||
Accepts either an expression (that evaluates to a sequence) or a subquery
|
||||
(hash map). In the expression case, also accepts an optional second argument
|
||||
that specifies the type of the array.
|
||||
|
||||
Produces either an `ARRAY[..]` or an `ARRAY(subquery)` expression.
|
||||
|
||||
In the expression case, produces `ARRAY[?, ?, ..]` for the elements of that
|
||||
sequence (as SQL parameters):
|
||||
|
||||
```clojure
|
||||
(sql/format-expr [:array (range 5)])
|
||||
;;=> ["ARRAY[?, ?, ?, ?, ?]" 0 1 2 3 4]
|
||||
(sql/format-expr [:array (range 3) :text])
|
||||
|
|
@ -45,7 +81,50 @@ In addition, the argument to `:array` is treated as a literal sequence of Clojur
|
|||
;;=> ["SELECT ARRAY[inline, (?, ?, ?)] AS arr" 1 2 3]
|
||||
```
|
||||
|
||||
## between
|
||||
In the subquery case, produces `ARRAY(subquery)`:
|
||||
|
||||
```clojure
|
||||
(sql/format {:select [[[:array {:select :* :from :table}] :arr]]})
|
||||
;;=> ["SELECT ARRAY(SELECT * FROM table) AS arr"]
|
||||
```
|
||||
|
||||
## at
|
||||
|
||||
If addition to dot navigation (for JSON) -- see the `.` and `.:.` syntax below --
|
||||
HoneySQL also supports bracket notation for JSON navigation.
|
||||
|
||||
The first argument to `:at` is treated as an expression that identifies
|
||||
the column, and subsequent arguments are treated as field names or array
|
||||
indices to navigate into that document.
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select [[[:at :col :field1 :field2]]]})
|
||||
["SELECT col.field1.field2"]
|
||||
user=> (sql/format {:select [[[:at :table.col 0 :field]]]})
|
||||
["SELECT table.col[0].field"]
|
||||
```
|
||||
|
||||
If you want an array index to be a parameter, use `:lift`:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select [[[:at :col [:lift 0] :field]]]})
|
||||
["SELECT col[?].field" 0]
|
||||
```
|
||||
|
||||
## at time zone
|
||||
|
||||
Accepts two arguments: an expression (assumed to be a date/time of some sort)
|
||||
and a time zone name or identifier (can be a string, a symbol, or a keyword):
|
||||
|
||||
```clojure
|
||||
(sql/format-expr [:at-time-zone [:now] :UTC])
|
||||
;;=> ["NOW() AT TIME ZONE 'UTC'"]
|
||||
```
|
||||
|
||||
The time zone name or identifier will be inlined (as a string) and therefore
|
||||
cannot be an expression.
|
||||
|
||||
## between and not-between
|
||||
|
||||
Accepts three arguments: an expression, a lower bound, and
|
||||
an upper bound:
|
||||
|
|
@ -53,6 +132,9 @@ an upper bound:
|
|||
```clojure
|
||||
(sql/format-expr [:between :id 1 100])
|
||||
;;=> ["id BETWEEN ? AND ?" 1 100]
|
||||
|
||||
(sql/format-expr [:not-between :id 1 100])
|
||||
;;=> ["id NOT BETWEEN ? AND ?" 1 100]
|
||||
```
|
||||
|
||||
## case
|
||||
|
|
@ -78,14 +160,44 @@ this using `:case-expr`:
|
|||
|
||||
## cast
|
||||
|
||||
A SQL CAST expression. Expects an expression and something
|
||||
A SQL `CAST` expression. Expects an expression and something
|
||||
that produces a SQL type:
|
||||
|
||||
```clojure
|
||||
(sql/format-expr [:cast :a :int])
|
||||
(sql/format [:cast :a :int])
|
||||
;;=> ["CAST(a AS INT)"]
|
||||
```
|
||||
|
||||
Quoting does not affect the type in a `CAST`, only the expression:
|
||||
|
||||
```clojure
|
||||
(sql/format [:cast :a :int] {:quoted true})
|
||||
;;=> ["CAST(\"a\" AS INT)"]
|
||||
```
|
||||
|
||||
A hyphen (`-`) in the type name becomes a space:
|
||||
|
||||
```clojure
|
||||
(sql/format [:cast :a :double-precision])
|
||||
;;=> ["CAST(a AS DOUBLE PRECISION)"]
|
||||
```
|
||||
|
||||
If you want an underscore in the type name, you have two choices:
|
||||
|
||||
```clojure
|
||||
(sql/format [:cast :a :some_type])
|
||||
;;=> ["CAST(a AS SOME_TYPE)"]
|
||||
```
|
||||
|
||||
or:
|
||||
|
||||
```clojure
|
||||
(sql/format [:cast :a :'some-type])
|
||||
;;=> ["CAST(a AS some_type)"]
|
||||
```
|
||||
|
||||
> Note: In HoneySQL 2.4.947 and earlier, the type name was incorrectly affected by the quoting feature, and a hyphen in a type name was incorrectly changed to underscore. This was corrected in 2.4.962.
|
||||
|
||||
## composite
|
||||
|
||||
Accepts any number of expressions and produces a composite
|
||||
|
|
@ -96,6 +208,23 @@ expression (comma-separated, wrapped in parentheses):
|
|||
;;=> ["(a, b, ?, x + ?)" "red" 1]
|
||||
```
|
||||
|
||||
This can be useful in a number of situations where you want a composite
|
||||
value, as above, or a composite based on or declaring columns names:
|
||||
|
||||
```clojure
|
||||
(sql/format {:select [[[:composite :a :b] :c]] :from :table})
|
||||
;;=> ["SELECT (a, b) AS c FROM table"]
|
||||
```
|
||||
|
||||
```clojure
|
||||
(sql/format {:update :table :set {:a :v.a}
|
||||
:from [[{:values [[1 2 3]
|
||||
[4 5 6]]}
|
||||
[:v [:composite :a :b :c]]]]
|
||||
:where [:and [:= :x :v.b] [:> :y :v.c]]})
|
||||
;;=> ["UPDATE table SET a = v.a FROM (VALUES (?, ?, ?), (?, ?, ?)) AS v (a, b, c) WHERE (x = v.b) AND (y > v.c)" 1 2 3 4 5 6]
|
||||
```
|
||||
|
||||
## distinct
|
||||
|
||||
Accepts a single expression and prefixes it with `DISTINCT `:
|
||||
|
|
@ -105,15 +234,23 @@ Accepts a single expression and prefixes it with `DISTINCT `:
|
|||
;;=> ["SELECT COUNT(DISTINCT status) AS n FROM table"]
|
||||
```
|
||||
|
||||
## dot .
|
||||
## dot . .:.
|
||||
|
||||
Accepts an expression and a field (or column) selection:
|
||||
Accepts an expression and one or more fields (or columns). Plain dot produces
|
||||
plain dotted selection:
|
||||
|
||||
```clojure
|
||||
(sql/format {:select [ [[:. :t :c]] [[:. :s :t :c]] ]})
|
||||
;;=> ["SELECT t.c, s.t.c"]
|
||||
```
|
||||
|
||||
Dot colon dot produces Snowflake-style dotted selection:
|
||||
|
||||
```clojure
|
||||
(sql/format {:select [ [[:.:. :t :c]] [[:.:. :s :t :c]] ]})
|
||||
;;=> ["SELECT t:c, s:t.c"]
|
||||
```
|
||||
|
||||
Can be used with `:nest` for field selection from composites:
|
||||
|
||||
```clojure
|
||||
|
|
@ -121,6 +258,9 @@ Can be used with `:nest` for field selection from composites:
|
|||
;;=> ["SELECT (v).*, (MYFUNC(x)).y"]
|
||||
```
|
||||
|
||||
See also [`get-in`](xtdb.md#object-navigation-expressions)
|
||||
and [`at`](#at) for additional path navigation functions.
|
||||
|
||||
## entity
|
||||
|
||||
Accepts a single keyword or symbol argument and produces a
|
||||
|
|
@ -192,10 +332,22 @@ FROM aa
|
|||
100]
|
||||
```
|
||||
|
||||
## ignore/respect nulls
|
||||
|
||||
Both of these accept a single argument -- an expression -- and
|
||||
renders that expression followed by `IGNORE NULLS` or `RESPECT NULLS`:
|
||||
|
||||
```clojure
|
||||
(sql/format-expr [:array_agg [:ignore-nulls :a]])
|
||||
;;=> ["ARRAY_AGG(a IGNORE NULLS)"]
|
||||
(sql/format-expr [:array_agg [:respect-nulls :a]])
|
||||
;;=> ["ARRAY_AGG(a RESPECT NULLS)"]
|
||||
```
|
||||
|
||||
## inline
|
||||
|
||||
Accepts a single argument and tries to render it as a
|
||||
SQL value directly in the formatted SQL string rather
|
||||
Accepts one or more arguments and tries to render them as a
|
||||
SQL values directly in the formatted SQL string rather
|
||||
than turning it into a positional parameter:
|
||||
* `nil` becomes `NULL`
|
||||
* keywords and symbols become upper case entities (with `-` replaced by space)
|
||||
|
|
@ -208,17 +360,73 @@ than turning it into a positional parameter:
|
|||
;;=> ["WHERE x = 'foo'"]
|
||||
```
|
||||
|
||||
If multiple arguments are provided, they are individually formatted as above
|
||||
and joined into a single SQL string with spaces:
|
||||
|
||||
```clojure
|
||||
(sql/format {:where [:= :x [:inline :DATE "2019-01-01"]]})
|
||||
;;=> ["WHERE x = DATE '2019-01-01'"]
|
||||
```
|
||||
|
||||
This is convenient for rendering DATE/TIME/TIMESTAMP literals in SQL.
|
||||
|
||||
If an argument is an expression, it is formatted as a regular SQL expression
|
||||
except that any parameters are inlined:
|
||||
|
||||
```clojure
|
||||
(sql/format {:where [:= :x [:inline [:date_add [:now] [:interval 30 :days]]]]})
|
||||
;;=> ["WHERE x = DATE_ADD(NOW(), INTERVAL 30 DAYS)"]
|
||||
```
|
||||
|
||||
In particular, that means that you can use `:inline` to inline a parameter
|
||||
value:
|
||||
|
||||
```clojure
|
||||
(sql/format {:where [:= :x [:inline :?foo]]} {:params {:foo "bar"}})
|
||||
;;=> ["WHERE x = 'bar'"]
|
||||
(sql/format {:where [:= :x [:inline [:param :foo]]]} {:params {:foo "bar"}})
|
||||
;;=> ["WHERE x = 'bar'"]
|
||||
```
|
||||
|
||||
## interval
|
||||
|
||||
Accepts two arguments: an expression and a keyword (or a symbol)
|
||||
that represents a time unit. Produces an `INTERVAL` expression:
|
||||
Accepts one or two arguments: either a string or an expression and
|
||||
a keyword (or a symbol) that represents a time unit.
|
||||
Produces an `INTERVAL` expression:
|
||||
|
||||
```clojure
|
||||
(sql/format-expr [:date_add [:now] [:interval 30 :days]])
|
||||
;;=> ["DATE_ADD(NOW(), INTERVAL ? DAYS)" 30]
|
||||
(sql/format-expr [:date_add [:now] [:interval "24 Hours"]])
|
||||
;;=> ["DATE_ADD(NOW(), INTERVAL '24 Hours')"]
|
||||
```
|
||||
|
||||
> Note: PostgreSQL has an `INTERVAL` data type which is unrelated to this syntax. In PostgreSQL, the closet equivalent would be `[:cast "30 days" :interval]` which will lift `"30 days"` out as a parameter. In DDL, for PostgreSQL, you can use `:interval` to produce the `INTERVAL` data type (without wrapping it in a vector).
|
||||
> Note: PostgreSQL also has an `INTERVAL` data type which is unrelated to this syntax. In PostgreSQL, the closet equivalent would be `[:cast "30 days" :interval]` which will lift `"30 days"` out as a parameter. In DDL, for PostgreSQL, you can use `:interval` to produce the `INTERVAL` data type (without wrapping it in a vector).
|
||||
|
||||
## join
|
||||
|
||||
Accepts a table name (or expression) followed by one or more join clauses.
|
||||
Produces a nested `JOIN` expression, typically used as the table expression of
|
||||
a `JOIN` clause.
|
||||
|
||||
```clojure
|
||||
(sql/format {:join [[[:join :tbl1 {:left-join [:tbl2 [:using :id]]}]]]})
|
||||
;;=> ["INNER JOIN (tbl1 LEFT JOIN tbl2 USING (id))"]
|
||||
```
|
||||
|
||||
An alias can be provided:
|
||||
|
||||
```clojure
|
||||
(sql/format {:join [[[:join [:tbl1 :t] {:left-join [:tbl2 [:using :id]]}]]]})
|
||||
;;=> ["INNER JOIN (tbl1 AS t LEFT JOIN tbl2 USING (id))"]
|
||||
```
|
||||
|
||||
To provide an expression, an extra level of `[...]` is needed:
|
||||
|
||||
```clojure
|
||||
(sql/format {:join [[[:join [[:make_thing 42] :t] {:left-join [:tbl2 [:using :id]]}]]]})
|
||||
;;=> ["INNER JOIN (MAKE_THING(?) AS t LEFT JOIN tbl2 USING (id))" 42]
|
||||
```
|
||||
|
||||
## lateral
|
||||
|
||||
|
|
|
|||
220
doc/xtdb.md
Normal file
220
doc/xtdb.md
Normal file
|
|
@ -0,0 +1,220 @@
|
|||
# XTDB Support
|
||||
|
||||
As of 2.6.1230, HoneySQL provides support for most of XTDB's SQL
|
||||
extensions, with additional support being added in subsequent releases.
|
||||
|
||||
For the most part, XTDB's SQL is based on
|
||||
[SQL:2011](https://en.wikipedia.org/wiki/SQL:2011), including the
|
||||
bitemporal features, but also includes a number of SQL extensions
|
||||
to support additional XTDB-specific features.
|
||||
|
||||
HoneySQL attempts to support all of these XTDB features in the core
|
||||
ANSI dialect, and this section documents most of those XTDB features.
|
||||
|
||||
For more details, see the XTDB documentation:
|
||||
* [SQL Overview](https://docs.xtdb.com/quickstart/sql-overview.html)
|
||||
* [SQL Queries](https://docs.xtdb.com/reference/main/sql/queries.html)
|
||||
* [SQL Transactions/DML](https://docs.xtdb.com/reference/main/sql/txs.html)
|
||||
|
||||
## Code Examples
|
||||
|
||||
The code examples herein assume:
|
||||
```clojure
|
||||
(refer-clojure :exclude '[update set])
|
||||
(require '[honey.sql :as sql]
|
||||
'[honey.sql.helpers :refer [select from where
|
||||
delete-from erase-from
|
||||
insert-into patch-into values
|
||||
records]])
|
||||
```
|
||||
|
||||
Clojure users can opt for the shorter `(require '[honey.sql :as sql] '[honey.sql.helpers :refer :all])` but this syntax is not available to ClojureScript users.
|
||||
|
||||
## `select` Variations
|
||||
|
||||
XTDB allows you to omit `SELECT` in a query. `SELECT *` is assumed if
|
||||
it is omitted. In HoneySQL, you can simply omit the `:select` clause
|
||||
from the DSL to achieve this.
|
||||
|
||||
```clojure
|
||||
user=> (sql/format '{select * from foo where (= status "active")})
|
||||
["SELECT * FROM foo WHERE status = ?" "active"]
|
||||
user=> (sql/format '{from foo where (= status "active")})
|
||||
["FROM foo WHERE status = ?" "active"]
|
||||
```
|
||||
|
||||
You can also `SELECT *` and then exclude columns and/or rename columns.
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select [[:* {:exclude :_id :rename [[:title, :name]]}]]})
|
||||
["SELECT * EXCLUDE _id RENAME title AS name"]
|
||||
user=> (sql/format '{select ((a.* {exclude _id})
|
||||
(b.* {rename ((title, name))}))
|
||||
from ((foo a))
|
||||
join ((bar b) (= a._id b.foo_id))})
|
||||
["SELECT a.* EXCLUDE _id, b.* RENAME title AS name FROM foo AS a INNER JOIN bar AS b ON a._id = b.foo_id"]
|
||||
```
|
||||
|
||||
`:exclude` can accept a single column, or a sequence of columns.
|
||||
`:rename` accepts a sequence of pairs (column name, new name).
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select [[:* {:exclude [:_id :upc]
|
||||
:rename [[:title, :name]
|
||||
[:price, :cost]]}]]})
|
||||
["SELECT * EXCLUDE (_id, upc) RENAME (title AS name, price AS cost)"]
|
||||
```
|
||||
|
||||
## Nested Sub-Queries
|
||||
|
||||
XTDB can produce structured results from `SELECT` queries containing
|
||||
sub-queries, using `NEST_ONE` and `NEST_MANY`. In HoneySQL, these are
|
||||
supported as regular function syntax in `:select` clauses.
|
||||
|
||||
```clojure
|
||||
user=> (sql/format '{select (a.*
|
||||
((nest_many {select * from bar where (= foo_id a._id)})
|
||||
b))
|
||||
from ((foo a))})
|
||||
["SELECT a.*, NEST_MANY (SELECT * FROM bar WHERE foo_id = a._id) AS b FROM foo AS a"]
|
||||
```
|
||||
|
||||
Remember that function calls in `:select` clauses need to be nested three
|
||||
levels of parentheses (brackets):
|
||||
`:select [:col-a [:col-b :alias-b] [[:fn-call :col-c] :alias-c]]`.
|
||||
|
||||
## `records` Clause
|
||||
|
||||
XTDB provides a `RECORDS` clause to specify a list of structured documents,
|
||||
similar to `VALUES` but specifically for documents rather than a collection
|
||||
of column values. HoneySQL supports a `:records` clauses and automatically
|
||||
lifts hash map values to parameters (rather than treating them as DSL fragments).
|
||||
You can inline a hash map to produce XTDB's inline document syntax.
|
||||
See also `insert` and `patch` below.
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:records [{:_id 1 :status "active"}]})
|
||||
["RECORDS ?" {:_id 1, :status "active"}]
|
||||
user=> (sql/format {:records [[:inline {:_id 1 :status "active"}]]})
|
||||
["RECORDS {_id: 1, status: 'active'}"]
|
||||
```
|
||||
|
||||
## `object` (`record`) Literals
|
||||
|
||||
While `RECORDS` exists in parallel to the `VALUES` clause, XTDB also provides
|
||||
a syntax to construct documents in other contexts in SQL, via the `OBJECT`
|
||||
literal syntax. `RECORD` is a synonym for `OBJECT`. HoneySQL supports both
|
||||
`:object` and `:record` as special syntax:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select [[[:object {:_id 1 :status "active"}]]]})
|
||||
["SELECT OBJECT (_id: 1, status: 'active')"]
|
||||
user=> (sql/format {:select [[[:record {:_id 1 :status "active"}]]]})
|
||||
["SELECT RECORD (_id: 1, status: 'active')"]
|
||||
```
|
||||
|
||||
A third option is to use `:inline` with a hash map:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select [[[:inline {:_id 1 :status "active"}]]]})
|
||||
["SELECT {_id: 1, status: 'active'}"]
|
||||
```
|
||||
|
||||
## Object Navigation Expressions
|
||||
|
||||
In order to deal with nested documents, XTDB provides syntax to navigate
|
||||
into them, via field names and/or array indices. HoneySQL supports this
|
||||
via the `:get-in` special syntax, intended to be familiar to Clojure users.
|
||||
|
||||
The first argument to `:get-in` is treated as an expression that produces
|
||||
the document, and subsequent arguments are treated as field names or array
|
||||
indices to navigate into that document.
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select [[[:get-in :doc :field1 :field2]]]})
|
||||
["SELECT (doc).field1.field2"]
|
||||
user=> (sql/format {:select [[[:get-in :table.col 0 :field]]]})
|
||||
["SELECT (table.col)[0].field"]
|
||||
```
|
||||
|
||||
If you want an array index to be a parameter, use `:lift`:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select [[[:get-in :doc [:lift 0] :field]]]})
|
||||
["SELECT (doc)[?].field" 0]
|
||||
```
|
||||
|
||||
## Temporal Queries
|
||||
|
||||
XTDB allows any query to be run in a temporal context via the `SETTING`
|
||||
clause (ahead of the `SELECT` clause). HoneySQL supports this via the
|
||||
`:setting` clause. It accepts a sequence of identifiers and expressions.
|
||||
An identifier ending in `-time` is assumed to be a temporal identifier
|
||||
(e.g., `:system-time` mapping to `SYSTEM_TIME`). Other identifiers are assumed to
|
||||
be regular SQL (so `-` is mapped to a space, e.g., `:as-of` mapping to `AS OF`).
|
||||
A timestamp literal, such as `DATE '2024-11-24'` can be specified in HoneySQL
|
||||
using `[:inline [:DATE "2024-11-24"]]` (note the literal case of `:DATE`
|
||||
to produce `DATE`).
|
||||
|
||||
See [XTDB's Top-level queries documentation](https://docs.xtdb.com/reference/main/sql/queries.html#_top_level_queries) for more details.
|
||||
|
||||
Here's one fairly complex example:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:setting [[:snapshot-time :to [:inline :DATE "2024-11-24"]]
|
||||
[:default :valid-time :to :between [:inline :DATE "2022"] :and [:inline :DATE "2023"]]]})
|
||||
["SETTING SNAPSHOT_TIME TO DATE '2024-11-24', DEFAULT VALID_TIME TO BETWEEN DATE '2022' AND DATE '2023'"]
|
||||
```
|
||||
|
||||
Table references (e.g., in a `FROM` clause) can also have temporal qualifiers.
|
||||
See [HoneySQL's `from` clause documentation](clause-reference.md#from) for
|
||||
examples of that, one of which is reproduced here:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:select [:username]
|
||||
:from [[:user :for :system-time :as-of [:inline "2019-08-01 15:23:00"]]]
|
||||
:where [:= :id 9]})
|
||||
["SELECT username FROM user FOR SYSTEM_TIME AS OF '2019-08-01 15:23:00' WHERE id = ?" 9]
|
||||
```
|
||||
|
||||
## `delete` and `erase`
|
||||
|
||||
In XTDB, `DELETE` is a temporal deletion -- the data remains in the database
|
||||
but is no longer visible in queries that don't specify a time range prior to
|
||||
the deletion. XTDB provides a similar `ERASE` operation that can permanently
|
||||
delete the data. HoneySQL supports `:erase-from` with the same syntax as
|
||||
`:delete-from`.
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:delete-from :foo :where [:= :status "inactive"]})
|
||||
["DELETE FROM foo WHERE status = ?" "inactive"]
|
||||
user=> (sql/format {:erase-from :foo :where [:= :status "inactive"]})
|
||||
["ERASE FROM foo WHERE status = ?" "inactive"]
|
||||
```
|
||||
|
||||
## `insert` and `patch`
|
||||
|
||||
XTDB supports `PATCH` as an upsert operation: it will update existing
|
||||
documents (via merging the new data) or insert new documents if they
|
||||
don't already exist. HoneySQL supports `:patch-into` with the same syntax
|
||||
as `:insert-into` with `:records`.
|
||||
|
||||
```clojure
|
||||
user=> (sql/format {:insert-into :foo
|
||||
:records [{:_id 1 :status "active"}]})
|
||||
["INSERT INTO foo RECORDS ?" {:_id 1, :status "active"}]
|
||||
user=> (sql/format {:patch-into :foo
|
||||
:records [{:_id 1 :status "active"}]})
|
||||
["PATCH INTO foo RECORDS ?" {:_id 1, :status "active"}]
|
||||
```
|
||||
|
||||
## `assert`
|
||||
|
||||
XTDB supports an `ASSERT` operation that will throw an exception if the
|
||||
asserted predicate is not true:
|
||||
|
||||
```clojure
|
||||
user=> (sql/format '{assert (not-exists {select 1 from users where (= email "james @example.com")})}
|
||||
:inline true)
|
||||
["ASSERT NOT EXISTS (SELECT 1 FROM users WHERE email = 'james @example.com')"]
|
||||
```
|
||||
1804
src/honey/sql.cljc
1804
src/honey/sql.cljc
File diff suppressed because it is too large
Load diff
|
|
@ -1,10 +1,14 @@
|
|||
;; copyright (c) 2020-2022 sean corfield, all rights reserved
|
||||
;; copyright (c) 2020-2025 sean corfield, all rights reserved
|
||||
|
||||
(ns honey.sql.helpers
|
||||
"Helper functions for the built-in clauses in honey.sql.
|
||||
|
||||
All helper functions are inherently variadic. Typical
|
||||
usage is threaded, like this:
|
||||
All helper functions are inherently variadic.
|
||||
|
||||
In general, `(helper :foo expr)` will produce `{:helper [:foo expr]}`,
|
||||
with a few exceptions: see the docstring of the helper function for details.
|
||||
|
||||
Typical usage is threaded, like this:
|
||||
|
||||
```
|
||||
(-> (select :a :b :c)
|
||||
|
|
@ -13,6 +17,16 @@
|
|||
(sql/format))
|
||||
```
|
||||
|
||||
or conditionally like this:
|
||||
|
||||
```
|
||||
(-> (select :a :b :c)
|
||||
(from :table)
|
||||
(cond->
|
||||
id (where [:= :id id]))
|
||||
(sql/format))
|
||||
```
|
||||
|
||||
Therefore all helpers can take an existing DSL expression
|
||||
as their first argument or, if the first argument is not
|
||||
a hash map, an empty DSL is assumed -- an empty hash map.
|
||||
|
|
@ -44,33 +58,26 @@
|
|||
bulk-collect-info [& args]
|
||||
|
||||
(as they are for all helper functions)."
|
||||
(:refer-clojure :exclude [filter for group-by into partition-by set update])
|
||||
(:refer-clojure :exclude [assert distinct filter for group-by into partition-by set update])
|
||||
(:require [clojure.core :as c]
|
||||
[honey.sql]))
|
||||
[honey.sql :as h]))
|
||||
|
||||
#?(:clj (set! *warn-on-reflection* true))
|
||||
|
||||
;; implementation helpers:
|
||||
|
||||
(defn- default-merge [current args]
|
||||
(let [current (cond
|
||||
(let [mdata (meta current)
|
||||
current (cond
|
||||
(nil? current) []
|
||||
(sequential? current) (vec current)
|
||||
:else [current])]
|
||||
(c/into current args)))
|
||||
|
||||
(defn- sym->kw
|
||||
"Given a symbol, produce a keyword, retaining the namespace
|
||||
qualifier, if any."
|
||||
[s]
|
||||
(if (symbol? s)
|
||||
(if-let [n (namespace s)]
|
||||
(keyword n (name s))
|
||||
(keyword (name s)))
|
||||
s))
|
||||
(c/into (with-meta current mdata) args)))
|
||||
|
||||
(defn- conjunction?
|
||||
[e]
|
||||
(and (ident? e)
|
||||
(contains? #{:and :or} (sym->kw e))))
|
||||
(contains? #{:and :or} (#'h/sym->kw e))))
|
||||
|
||||
(defn- simplify-logic
|
||||
"For Boolean expressions, simplify the logic to make
|
||||
|
|
@ -82,11 +89,11 @@
|
|||
[e]
|
||||
(if (= 1 (count (rest e)))
|
||||
(fnext e)
|
||||
(let [conjunction (sym->kw (first e))]
|
||||
(let [conjunction (#'h/sym->kw (first e))]
|
||||
(reduce (fn [acc e]
|
||||
(if (and (sequential? e)
|
||||
(conjunction? (first e))
|
||||
(= conjunction (sym->kw (first e))))
|
||||
(= conjunction (#'h/sym->kw (first e))))
|
||||
(c/into acc (rest e))
|
||||
(conj acc e)))
|
||||
[conjunction]
|
||||
|
|
@ -130,11 +137,26 @@
|
|||
:having #'conjunction-merge})
|
||||
|
||||
(defn- helper-merge [data k args]
|
||||
(if-let [merge-fn (special-merges k)]
|
||||
(if-some [clause (merge-fn (get data k) args)]
|
||||
(assoc data k clause)
|
||||
data)
|
||||
(clojure.core/update data k default-merge args)))
|
||||
(let [k' (#'h/sym->kw k)
|
||||
k (#'h/kw->sym k)
|
||||
d (get data k)
|
||||
d' (get data k')
|
||||
mf (special-merges k')
|
||||
mf' (or mf default-merge)]
|
||||
(cond (some? d)
|
||||
(if-some [clause (mf' d args)]
|
||||
(assoc data k clause)
|
||||
data)
|
||||
(some? d')
|
||||
(if-some [clause (mf' d' args)]
|
||||
(assoc data k' clause)
|
||||
data)
|
||||
mf
|
||||
(if-some [clause (mf nil args)]
|
||||
(assoc data k' clause)
|
||||
data)
|
||||
:else
|
||||
(c/update data k' default-merge args))))
|
||||
|
||||
(defn- generic [k args]
|
||||
(if (map? (first args))
|
||||
|
|
@ -310,14 +332,23 @@
|
|||
(generic :with-columns args)))
|
||||
|
||||
(defn create-view
|
||||
"Accepts a single view name to create.
|
||||
" Accepts a single view name to create.
|
||||
|
||||
(-> (create-view :cities)
|
||||
(select :*) (from :city))"
|
||||
(-> (create-view :cities)
|
||||
(select :*) (from :city)) "
|
||||
{:arglists '([view])}
|
||||
[& args]
|
||||
(generic :create-view args))
|
||||
|
||||
(defn create-or-replace-view
|
||||
"Accepts a single view name to create.
|
||||
|
||||
(-> (create-or-replace-view :cities)
|
||||
(select :*) (from :city))"
|
||||
{:arglists '([view])}
|
||||
[& args]
|
||||
(generic :create-or-replace-view args))
|
||||
|
||||
(defn create-materialized-view
|
||||
"Accepts a single view name to create.
|
||||
|
||||
|
|
@ -356,6 +387,25 @@
|
|||
[& views]
|
||||
(generic :refresh-materialized-view views))
|
||||
|
||||
(defn create-index
|
||||
"Accepts an index spexification and a column specification. The column
|
||||
specification consists of table name and one or more columns.
|
||||
|
||||
(create-index :name-of-idx [:table :col])
|
||||
(create-index :name-of-idx [:table :col1 :col2])
|
||||
(create-index [:unique :name-of-idx] [:table :col])
|
||||
|
||||
PostgreSQL also supports :if-not-exists and expressions instead of columns.
|
||||
|
||||
(create-index [:name-of-idx :if-not-exists] [:table :%lower.col])"
|
||||
[& args]
|
||||
(generic :create-index args))
|
||||
|
||||
(defn setting
|
||||
"Accepts one or more time settings for a query."
|
||||
[& args]
|
||||
(generic :setting args))
|
||||
|
||||
(defn with
|
||||
"Accepts one or more CTE definitions.
|
||||
|
||||
|
|
@ -402,6 +452,14 @@
|
|||
[& clauses]
|
||||
(generic :except-all (cons {} clauses)))
|
||||
|
||||
(defn assert
|
||||
"Accepts an expression (predicate).
|
||||
|
||||
Produces: ASSERT expression"
|
||||
{:arglists '([expr])}
|
||||
[& args]
|
||||
(generic-1 :assert args))
|
||||
|
||||
(defn select
|
||||
"Accepts any number of column names, or column/alias
|
||||
pairs, or SQL expressions (optionally aliased):
|
||||
|
|
@ -455,6 +513,32 @@
|
|||
[& args]
|
||||
(generic :select-distinct-top args))
|
||||
|
||||
(defn records
|
||||
"Produces RECORDS {...}, {...}, ...
|
||||
Like `values` so it accepts a collection of maps."
|
||||
[& args]
|
||||
(generic-1 :records args))
|
||||
|
||||
(defn distinct
|
||||
"Like `select-distinct` but produces DISTINCT..."
|
||||
[& args]
|
||||
(generic-1 :distinct args))
|
||||
|
||||
(defn expr
|
||||
"Like `distinct` but produces ... (i.e., just the expression that follows)."
|
||||
[& args]
|
||||
(generic-1 :expr args))
|
||||
|
||||
(defn exclude
|
||||
"Accepts one or more column names to exclude from a select list."
|
||||
[& args]
|
||||
(generic :exclude args))
|
||||
|
||||
(defn rename
|
||||
"Accepts one or more column names with aliases to rename in a select list."
|
||||
[& args]
|
||||
(generic :rename args))
|
||||
|
||||
(defn into
|
||||
"Accepts table name, optionally followed a database name."
|
||||
{:arglists '([table] [table dbname])}
|
||||
|
|
@ -468,6 +552,14 @@
|
|||
[& args]
|
||||
(generic :bulk-collect-into args))
|
||||
|
||||
(defn- stuff-into [k args]
|
||||
(let [[data & args :as args']
|
||||
(if (map? (first args)) args (cons {} args))
|
||||
[table cols statement] args]
|
||||
(if (and (sequential? cols) (map? statement))
|
||||
(generic k [data [table cols] statement])
|
||||
(generic k args'))))
|
||||
|
||||
(defn insert-into
|
||||
"Accepts a table name or a table/alias pair. That
|
||||
can optionally be followed by a collection of
|
||||
|
|
@ -483,12 +575,20 @@
|
|||
(-> (select :*) (from :other)))"
|
||||
{:arglists '([table] [table cols] [table statement] [table cols statement])}
|
||||
[& args]
|
||||
(let [[data & args :as args']
|
||||
(if (map? (first args)) args (cons {} args))
|
||||
[table cols statement] args]
|
||||
(if (and (sequential? cols) (map? statement))
|
||||
(generic :insert-into [data [table cols] statement])
|
||||
(generic :insert-into args'))))
|
||||
(stuff-into :insert-into args))
|
||||
|
||||
(defn patch-into
|
||||
"Accepts a table name or a table/alias pair. That
|
||||
can optionally be followed by a collection of
|
||||
column names. That can optionally be followed by
|
||||
a (select) statement clause.
|
||||
|
||||
The arguments are identical to insert-into.
|
||||
The PATCH INTO statement is only supported by
|
||||
XTDB."
|
||||
{:arglists '([table] [table cols] [table statement] [table cols statement])}
|
||||
[& args]
|
||||
(stuff-into :patch-into args))
|
||||
|
||||
(defn replace-into
|
||||
"Accepts a table name or a table/alias pair. That
|
||||
|
|
@ -501,7 +601,7 @@
|
|||
MySQL and SQLite."
|
||||
{:arglists '([table] [table cols] [table statement] [table cols statement])}
|
||||
[& args]
|
||||
(apply insert-into args))
|
||||
(stuff-into :replace-into args))
|
||||
|
||||
(defn update
|
||||
"Accepts either a table name or a table/alias pair.
|
||||
|
|
@ -529,6 +629,15 @@
|
|||
[& args]
|
||||
(generic :delete-from args))
|
||||
|
||||
(defn erase-from
|
||||
"For erasing (hard delete) from a single table (XTDB).
|
||||
Accepts a single table name to erase from.
|
||||
|
||||
(-> (erase-from :films) (where [:= :id 1]))"
|
||||
{:arglists '([table])}
|
||||
[& args]
|
||||
(generic :erase-from args))
|
||||
|
||||
(defn truncate
|
||||
"Accepts a single table name to truncate."
|
||||
{:arglists '([table])}
|
||||
|
|
@ -869,7 +978,8 @@
|
|||
|
||||
(defn on-conflict
|
||||
"Accepts zero or more SQL entities (keywords or symbols),
|
||||
optionally followed by a single SQL clause (hash map)."
|
||||
optionally followed by a single SQL clause (`{:where <condition>}`).
|
||||
Ex.: `(on-conflict :mom :dad {:where [:= :race \"human\"]}`"
|
||||
{:arglists '([column* where-clause])}
|
||||
[& args]
|
||||
(generic :on-conflict args))
|
||||
|
|
@ -989,6 +1099,59 @@
|
|||
[& args]
|
||||
(c/into [:within-group] args))
|
||||
|
||||
;; nrql-specific helpers:
|
||||
|
||||
(defn facet
|
||||
"Accepts any number of column names, or column/alias
|
||||
pairs, or SQL expressions (optionally aliased):
|
||||
|
||||
(facet :id [:foo :bar] [[:max :quux]])
|
||||
|
||||
Produces: FACET id, foo AS bar, MAX(quux)"
|
||||
[& args]
|
||||
(generic :facet args))
|
||||
|
||||
(defn since
|
||||
"Accepts a time interval such as:
|
||||
|
||||
(since 2 :days :ago)
|
||||
|
||||
Produces: SINCE 2 DAYS AGO"
|
||||
[& args]
|
||||
(generic :since args))
|
||||
|
||||
(defn until
|
||||
"Accepts a time interval such as:
|
||||
|
||||
(until 1 :month :ago)
|
||||
|
||||
Produces: UNTIL 1 MONTH AGO"
|
||||
[& args]
|
||||
(generic :until args))
|
||||
|
||||
(defn compare-with
|
||||
"Accepts a time interval such as:
|
||||
|
||||
(compare-with 1 :week :ago)
|
||||
|
||||
Produces: COMPARE WITH 1 WEEK AGO"
|
||||
[& args]
|
||||
(generic :compare-with args))
|
||||
|
||||
(defn timeseries
|
||||
"Accepts a time interval such as:
|
||||
|
||||
(timeseries 1 :day)
|
||||
|
||||
or:
|
||||
|
||||
(timeseries :auto)
|
||||
|
||||
Produces: TIMESERIES 1 DAY
|
||||
Or: TIMESERIES AUTO"
|
||||
[& args]
|
||||
(generic :timeseries args))
|
||||
|
||||
;; this helper is intended to ease the migration from nilenso:
|
||||
(defn upsert
|
||||
"Provided purely to ease migration from nilenso/honeysql-postgres
|
||||
|
|
@ -1069,16 +1232,11 @@
|
|||
[k args]
|
||||
(generic-1 k args))
|
||||
|
||||
#?(:clj
|
||||
(do
|
||||
;; #409 this assert is only valid when :doc metadata is not elided:
|
||||
(when (-> #'generic-helper-unary meta :doc)
|
||||
;; ensure #295 stays true (all public functions have docstring):
|
||||
(assert (empty? (->> (ns-publics *ns*) (vals) (c/filter (comp not :doc meta))))))
|
||||
;; ensure all public functions match clauses:
|
||||
(assert (= (clojure.core/set (conj @#'honey.sql/default-clause-order
|
||||
:composite :filter :lateral :over :within-group
|
||||
:upsert
|
||||
:generic-helper-variadic :generic-helper-unary))
|
||||
(clojure.core/set (conj (map keyword (keys (ns-publics *ns*)))
|
||||
:nest :raw))))))
|
||||
(comment
|
||||
(-> (delete-from :table)
|
||||
(where [:in (composite :first :second)
|
||||
[(composite 1 2) (composite 2 1)]])
|
||||
(h/format))
|
||||
(-> (select [:%count.* :total]) (from :foo) h/format)
|
||||
(-> (select [[:count :*] :total]) (from :foo) h/format)
|
||||
)
|
||||
|
|
|
|||
|
|
@ -1,4 +1,4 @@
|
|||
;; copyright (c) 2022 sean corfield, all rights reserved
|
||||
;; copyright (c) 2022-2024 sean corfield, all rights reserved
|
||||
|
||||
(ns honey.sql.pg-ops
|
||||
"Register all the PostgreSQL JSON/JSONB operators
|
||||
|
|
@ -25,6 +25,8 @@
|
|||
(:refer-clojure :exclude [-> ->> -])
|
||||
(:require [honey.sql :as sql]))
|
||||
|
||||
#?(:clj (set! *warn-on-reflection* true))
|
||||
|
||||
;; see https://www.postgresql.org/docs/current/functions-json.html
|
||||
|
||||
(def ->
|
||||
|
|
@ -49,13 +51,16 @@
|
|||
(def -
|
||||
"The - operator:
|
||||
- text value: deletes a key (and its value) from a JSON object, or matching string value(s) from a JSON array
|
||||
- text[] array value: as above, but for all the provided keys
|
||||
- int value: deletes the array element with specified index (negative integers count from the end)"
|
||||
:-)
|
||||
(def hash- "The #- operator - deletes the field or array element at the specified path, where path elements can be either field keys or array indexes." :#-)
|
||||
(def at? "The @? operator - does JSON path return any item for the specified JSON value?" (keyword "@?"))
|
||||
(def atat
|
||||
"The @@ operator - returns the result of a JSON path predicate check for the specified JSON value.
|
||||
Only the first item of the result is taken into account. If the result is not Boolean, then NULL is returned."
|
||||
"The @@ operator:
|
||||
- returns the result of a JSON path predicate check for the specified JSON value. Only the first item of the result is taken into account.
|
||||
If the result is not Boolean, then NULL is returned.
|
||||
- checks if a text search vector (or a text value implicitly converted to a text search vector) matches a text search query. Returns a Boolean."
|
||||
(keyword "@@"))
|
||||
|
||||
(def tilde "The case-sensitive regex match operator." (keyword "~"))
|
||||
|
|
|
|||
|
|
@ -1,8 +1,10 @@
|
|||
;; copyright (c) 2022 sean corfield, all rights reserved
|
||||
;; copyright (c) 2022-2024 sean corfield, all rights reserved
|
||||
|
||||
(ns honey.sql.protocols
|
||||
"InlineValue -- a protocol that defines how to inline
|
||||
values; (sqlize x) produces a SQL string for x.")
|
||||
|
||||
#?(:clj (set! *warn-on-reflection* true))
|
||||
|
||||
(defprotocol InlineValue :extend-via-metadata true
|
||||
(sqlize [this] "Render value inline in a SQL string."))
|
||||
|
|
|
|||
109
src/honey/sql/util.cljc
Normal file
109
src/honey/sql/util.cljc
Normal file
|
|
@ -0,0 +1,109 @@
|
|||
(ns honey.sql.util
|
||||
"Utility functions for the main honey.sql namespace."
|
||||
(:refer-clojure :exclude [str])
|
||||
(:require clojure.string))
|
||||
|
||||
#?(:clj (set! *warn-on-reflection* true))
|
||||
|
||||
(defn str
|
||||
"More efficient implementation of `clojure.core/str` because it has more
|
||||
non-variadic arities. Optimization is Clojure-only, on other platforms it
|
||||
reverts back to `clojure.core/str`."
|
||||
(^String [] "")
|
||||
(^String [^Object a]
|
||||
#?(:clj (if (nil? a) "" (.toString a))
|
||||
:default (clojure.core/str a)))
|
||||
(^String [^Object a, ^Object b]
|
||||
#?(:clj (if (nil? a)
|
||||
(str b)
|
||||
(if (nil? b)
|
||||
(.toString a)
|
||||
(.concat (.toString a) (.toString b))))
|
||||
:default (clojure.core/str a b)))
|
||||
(^String [a b c]
|
||||
#?(:clj (let [sb (StringBuilder.)]
|
||||
(.append sb (str a))
|
||||
(.append sb (str b))
|
||||
(.append sb (str c))
|
||||
(.toString sb))
|
||||
:default (clojure.core/str a b c)))
|
||||
(^String [a b c d]
|
||||
#?(:clj (let [sb (StringBuilder.)]
|
||||
(.append sb (str a))
|
||||
(.append sb (str b))
|
||||
(.append sb (str c))
|
||||
(.append sb (str d))
|
||||
(.toString sb))
|
||||
:default (clojure.core/str a b c d)))
|
||||
(^String [a b c d e]
|
||||
#?(:clj (let [sb (StringBuilder.)]
|
||||
(.append sb (str a))
|
||||
(.append sb (str b))
|
||||
(.append sb (str c))
|
||||
(.append sb (str d))
|
||||
(.append sb (str e))
|
||||
(.toString sb))
|
||||
:default (clojure.core/str a b c d e)))
|
||||
(^String [a b c d e & more]
|
||||
#?(:clj (let [sb (StringBuilder.)]
|
||||
(.append sb (str a))
|
||||
(.append sb (str b))
|
||||
(.append sb (str c))
|
||||
(.append sb (str d))
|
||||
(.append sb (str e))
|
||||
(run! #(.append sb (str %)) more)
|
||||
(.toString sb))
|
||||
:default (apply clojure.core/str a b c d e more))))
|
||||
|
||||
(defn join
|
||||
"More efficient implementation of `clojure.string/join`. May accept a transducer
|
||||
`xform` to perform operations on each element before combining them together
|
||||
into a string. Clojure-only, delegates to `clojure.string/join` on other
|
||||
platforms."
|
||||
([separator coll] (join separator identity coll))
|
||||
([separator xform coll]
|
||||
#?(:clj
|
||||
(let [sb (StringBuilder.)
|
||||
sep (str separator)]
|
||||
(transduce xform
|
||||
(fn
|
||||
([] false)
|
||||
([_] (.toString sb))
|
||||
([add-sep? x]
|
||||
(when add-sep? (.append sb sep))
|
||||
(.append sb (str x))
|
||||
true))
|
||||
false coll))
|
||||
|
||||
:default
|
||||
(clojure.string/join separator (transduce xform conj [] coll)))))
|
||||
|
||||
(defn split-by-separator
|
||||
"More efficient implementation of `clojure.string/split` for cases when a
|
||||
literal string (not regex) is used as a separator, and for cases where the
|
||||
separator is not present in the haystack at all."
|
||||
[s sep]
|
||||
(loop [start 0, res []]
|
||||
(if-some [sep-idx (clojure.string/index-of s sep start)]
|
||||
(let [sep-idx (long sep-idx)]
|
||||
(recur (inc sep-idx) (conj res (subs s start sep-idx))))
|
||||
(if (= start 0)
|
||||
;; Fastpath - zero separators in s
|
||||
[s]
|
||||
(conj res (subs s start))))))
|
||||
|
||||
(defn into*
|
||||
"An extension of `clojure.core/into` that accepts multiple \"from\" arguments.
|
||||
Doesn't support `xform`."
|
||||
([to from1] (into* to from1 nil nil nil))
|
||||
([to from1 from2] (into* to from1 from2 nil nil))
|
||||
([to from1 from2 from3] (into* to from1 from2 from3 nil))
|
||||
([to from1 from2 from3 from4]
|
||||
(if (or from1 from2 from3 from4)
|
||||
(as-> (transient to) to'
|
||||
(reduce conj! to' from1)
|
||||
(reduce conj! to' from2)
|
||||
(reduce conj! to' from3)
|
||||
(reduce conj! to' from4)
|
||||
(persistent! to'))
|
||||
to)))
|
||||
|
|
@ -1,26 +0,0 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
<name>honeysql</name>
|
||||
<description>SQL as Clojure data structures.</description>
|
||||
<url>https://github.com/seancorfield/honeysql</url>
|
||||
<licenses>
|
||||
<license>
|
||||
<name>Eclipse Public License</name>
|
||||
<url>http://www.eclipse.org/legal/epl-v10.html</url>
|
||||
</license>
|
||||
</licenses>
|
||||
<developers>
|
||||
<developer>
|
||||
<name>Sean Corfield</name>
|
||||
</developer>
|
||||
<developer>
|
||||
<name>Justin Kramer</name>
|
||||
</developer>
|
||||
</developers>
|
||||
<scm>
|
||||
<url>https://github.com/seancorfield/honeysql</url>
|
||||
<connection>scm:git:git://github.com/seancorfield/honeysql.git</connection>
|
||||
<developerConnection>scm:git:ssh://git@github.com/seancorfield/honeysql.git</developerConnection>
|
||||
</scm>
|
||||
</project>
|
||||
|
|
@ -1,4 +1,4 @@
|
|||
;; copyright (c) 2022 sean corfield, all rights reserved
|
||||
;; copyright (c) 2022-2024 sean corfield, all rights reserved
|
||||
|
||||
(ns honey.bigquery-test
|
||||
(:refer-clojure :exclude [format])
|
||||
|
|
|
|||
|
|
@ -1,4 +1,4 @@
|
|||
;; copyright (c) 2022 sean corfield, all rights reserved
|
||||
;; copyright (c) 2022-2024 sean corfield, all rights reserved
|
||||
|
||||
(ns honey.cache-test
|
||||
(:refer-clojure :exclude [format group-by])
|
||||
|
|
|
|||
|
|
@ -1,12 +1,19 @@
|
|||
;; copyright (c) 2023 sean corfield, all rights reserved
|
||||
;; copyright (c) 2023-2025 sean corfield, all rights reserved
|
||||
|
||||
(ns honey.ops-test
|
||||
(:refer-clojure :exclude [format])
|
||||
(:require [clojure.test :refer [deftest is]]
|
||||
[honey.sql :as sut]
|
||||
[honey.sql :as sql]))
|
||||
[honey.sql :as sut]))
|
||||
|
||||
(deftest issue-454
|
||||
(is (= ["SELECT a - b - c AS x"]
|
||||
(-> {:select [[[:- :a :b :c] :x]]}
|
||||
(sql/format)))))
|
||||
(sut/format)))))
|
||||
|
||||
(deftest issue-566
|
||||
(is (= ["SELECT * FROM table WHERE a IS DISTINCT FROM b"]
|
||||
(-> {:select :* :from :table :where [:is-distinct-from :a :b]}
|
||||
(sut/format))))
|
||||
(is (= ["SELECT * FROM table WHERE a IS NOT DISTINCT FROM b"]
|
||||
(-> {:select :* :from :table :where [:is-not-distinct-from :a :b]}
|
||||
(sut/format)))))
|
||||
|
|
|
|||
|
|
@ -1,24 +1,44 @@
|
|||
;; copyright (c) 2020-2022 sean corfield, all rights reserved
|
||||
;; copyright (c) 2020-2024 sean corfield, all rights reserved
|
||||
|
||||
(ns honey.sql.helpers-test
|
||||
(:refer-clojure :exclude [filter for group-by partition-by set update])
|
||||
(:require [clojure.test :refer [deftest is testing]]
|
||||
#_{:clj-kondo/ignore [:unused-namespace]}
|
||||
(:require [clojure.core :as c]
|
||||
[clojure.test :refer [deftest is testing]]
|
||||
[honey.sql :as sql]
|
||||
[honey.sql.helpers :as h
|
||||
:refer [add-column add-index alter-table columns create-table create-table-as create-view
|
||||
create-materialized-view drop-view drop-materialized-view
|
||||
:refer [add-column alter-table columns create-table create-table-as create-view
|
||||
create-materialized-view
|
||||
create-index
|
||||
bulk-collect-into
|
||||
cross-join do-update-set drop-column drop-index drop-table
|
||||
cross-join do-update-set drop-column drop-table
|
||||
filter from full-join
|
||||
group-by having insert-into
|
||||
join-by join lateral left-join limit offset on-conflict
|
||||
group-by having insert-into replace-into
|
||||
join-by join left-join limit offset on-conflict
|
||||
on-duplicate-key-update
|
||||
order-by over partition-by refresh-materialized-view
|
||||
rename-column rename-table returning right-join
|
||||
select select-distinct select-top select-distinct-top
|
||||
returning right-join
|
||||
select select-distinct select-top
|
||||
values where window with with-columns
|
||||
with-data within-group]]))
|
||||
|
||||
#?(:clj
|
||||
(deftest helpers-are-complete
|
||||
(let [helpers-ns (find-ns 'honey.sql.helpers)]
|
||||
(testing "all public helpers have docstrings"
|
||||
;; #409 this assert is only valid when :doc metadata is not elided:
|
||||
(when (-> #'h/generic-helper-unary meta :doc)
|
||||
;; ensure #295 stays true (all public functions have docstring):
|
||||
(is (= [] (->> (ns-publics helpers-ns) (vals) (c/filter (comp not :doc meta)))))))
|
||||
(testing "all clauses have public helpers"
|
||||
;; ensure all public functions match clauses:
|
||||
(is (= (c/set (conj @#'honey.sql/default-clause-order
|
||||
:composite :filter :lateral :over :within-group
|
||||
:upsert
|
||||
:generic-helper-variadic :generic-helper-unary))
|
||||
(c/set (conj (map keyword (keys (ns-publics helpers-ns)))
|
||||
:nest :raw))))))))
|
||||
|
||||
(deftest test-select
|
||||
(testing "large helper expression"
|
||||
(let [m1 (-> (with [:cte (-> (select :*)
|
||||
|
|
@ -335,7 +355,15 @@
|
|||
{:params {:ids values} :numbered true})))))))
|
||||
|
||||
(deftest test-case
|
||||
(is (= ["SELECT CASE WHEN foo < ? THEN ? WHEN (foo > ?) AND ((foo MOD ?) = ?) THEN foo / ? ELSE ? END FROM bar"
|
||||
(is (= ["SELECT CASE WHEN foo < ? THEN ? WHEN (foo > ?) AND ((foo % ?) = ?) THEN foo / ? ELSE ? END FROM bar"
|
||||
0 -1 0 2 0 2 0]
|
||||
(sql/format
|
||||
{:select [[[:case
|
||||
[:< :foo 0] -1
|
||||
[:and [:> :foo 0] [:= [:% :foo 2] 0]] [:/ :foo 2]
|
||||
:else 0]]]
|
||||
:from [:bar]})))
|
||||
(is (= ["SELECT CASE WHEN foo < ? THEN ? WHEN (foo > ?) AND (MOD(foo, ?) = ?) THEN foo / ? ELSE ? END FROM bar"
|
||||
0 -1 0 2 0 2 0]
|
||||
(sql/format
|
||||
{:select [[[:case
|
||||
|
|
@ -524,6 +552,33 @@
|
|||
" MAX(salary) OVER w AS MaxSalary"
|
||||
" FROM employee"
|
||||
" WINDOW w AS (PARTITION BY department)")]))
|
||||
;; multiple window tests
|
||||
(is (= (-> (select :id
|
||||
(over [[:avg :salary] (-> (partition-by :department) (order-by :designation)) :Average]
|
||||
[[:max :salary] :w :MaxSalary]))
|
||||
(from :employee)
|
||||
(window :w (partition-by :department))
|
||||
(window :x (partition-by :salary))
|
||||
sql/format)
|
||||
[(str "SELECT id,"
|
||||
" AVG(salary) OVER (PARTITION BY department ORDER BY designation ASC) AS Average,"
|
||||
" MAX(salary) OVER w AS MaxSalary"
|
||||
" FROM employee"
|
||||
" WINDOW w AS (PARTITION BY department)"
|
||||
", x AS (PARTITION BY salary)")]))
|
||||
(is (= (-> (select :id
|
||||
(over [[:avg :salary] (-> (partition-by :department) (order-by :designation)) :Average]
|
||||
[[:max :salary] :w :MaxSalary]))
|
||||
(from :employee)
|
||||
(window :w (partition-by :department)
|
||||
:x (partition-by :salary))
|
||||
sql/format)
|
||||
[(str "SELECT id,"
|
||||
" AVG(salary) OVER (PARTITION BY department ORDER BY designation ASC) AS Average,"
|
||||
" MAX(salary) OVER w AS MaxSalary"
|
||||
" FROM employee"
|
||||
" WINDOW w AS (PARTITION BY department)"
|
||||
", x AS (PARTITION BY salary)")]))
|
||||
;; test nil / empty window function clause:
|
||||
(is (= (-> (select :id
|
||||
(over [[:avg :salary] {} :Average]
|
||||
|
|
@ -553,6 +608,24 @@
|
|||
(where [:= :metroflag "y"])
|
||||
(with-data false)))
|
||||
["CREATE TABLE IF NOT EXISTS metro AS SELECT * FROM cities WHERE metroflag = ? WITH NO DATA" "y"]))
|
||||
(is (= (sql/format (-> (create-table-as :metro :or-replace)
|
||||
(select :*)
|
||||
(from :cities)
|
||||
(where [:= :metroflag "y"])
|
||||
(with-data false)))
|
||||
["CREATE OR REPLACE TABLE metro AS SELECT * FROM cities WHERE metroflag = ? WITH NO DATA" "y"]))
|
||||
(is (= (sql/format (-> (create-table-as :temp :metro :if-not-exists)
|
||||
(select :*)
|
||||
(from :cities)
|
||||
(where [:= :metroflag "y"])
|
||||
(with-data false)))
|
||||
["CREATE TEMP TABLE IF NOT EXISTS metro AS SELECT * FROM cities WHERE metroflag = ? WITH NO DATA" "y"]))
|
||||
(is (= (sql/format (-> (create-table-as :temp :metro :or-replace)
|
||||
(select :*)
|
||||
(from :cities)
|
||||
(where [:= :metroflag "y"])
|
||||
(with-data false)))
|
||||
["CREATE OR REPLACE TEMP TABLE metro AS SELECT * FROM cities WHERE metroflag = ? WITH NO DATA" "y"]))
|
||||
(is (= (sql/format (-> (create-materialized-view :metro :if-not-exists)
|
||||
(select :*)
|
||||
(from :cities)
|
||||
|
|
@ -569,6 +642,16 @@
|
|||
[(str "CREATE TABLE IF NOT EXISTS metro"
|
||||
" (foo, bar, baz) TABLESPACE quux"
|
||||
" AS SELECT * FROM cities WHERE metroflag = ? WITH NO DATA") "y"]))
|
||||
(is (= (sql/format (-> (create-table-as :metro :or-replace
|
||||
(columns :foo :bar :baz)
|
||||
[:tablespace [:entity :quux]])
|
||||
(select :*)
|
||||
(from :cities)
|
||||
(where [:= :metroflag "y"])
|
||||
(with-data false)))
|
||||
[(str "CREATE OR REPLACE TABLE metro"
|
||||
" (foo, bar, baz) TABLESPACE quux"
|
||||
" AS SELECT * FROM cities WHERE metroflag = ? WITH NO DATA") "y"]))
|
||||
(is (= (sql/format (-> (create-materialized-view :metro :if-not-exists
|
||||
(columns :foo :bar :baz)
|
||||
[:tablespace [:entity :quux]])
|
||||
|
|
@ -753,7 +836,10 @@
|
|||
["INSERT INTO transport (id, name) SELECT * FROM cars"]))
|
||||
;; three arguments with an alias and columns:
|
||||
(is (= (sql/format (insert-into '(transport t) '(id, name) '{select (*) from (cars)}))
|
||||
["INSERT INTO transport AS t (id, name) SELECT * FROM cars"])))
|
||||
["INSERT INTO transport AS t (id, name) SELECT * FROM cars"]))
|
||||
;; and again with replace-into:
|
||||
(is (= (sql/format (replace-into '(transport t) '(id, name) '{select (*) from (cars)}))
|
||||
["REPLACE INTO transport AS t (id, name) SELECT * FROM cars"])))
|
||||
|
||||
;; these tests are adapted from Cam Saul's PR #283
|
||||
|
||||
|
|
@ -913,3 +999,62 @@
|
|||
(where false)))
|
||||
(is (= ["SELECT * FROM table WHERE FALSE"]
|
||||
(sql/format {:select [:*] :from [:table] :where false})))))
|
||||
|
||||
(deftest issue-505
|
||||
(testing "where should merge symbols/keywords correctly"
|
||||
(is (= '{where [:and (= a 1) [:= :b 2]]}
|
||||
(-> '{where (= a 1)}
|
||||
(where [:= :b 2]))))
|
||||
(is (= '{where (= a 1)}
|
||||
(-> '{where (= a 1)}
|
||||
(where))))
|
||||
(is (= '{:where [:and (= a 1) [:= :b 2]]}
|
||||
(-> '{:where (= a 1)}
|
||||
(where [:= :b 2]))))
|
||||
(is (= '{:where (= a 1)}
|
||||
(-> '{:where (= a 1)}
|
||||
(where))))
|
||||
(is (= '{:where [:= :b 2]}
|
||||
(-> '{}
|
||||
(where [:= :b 2]))))
|
||||
(is (= '{}
|
||||
(-> '{}
|
||||
(where))))))
|
||||
|
||||
(deftest test-create-index
|
||||
(testing "create index, commonly supported features"
|
||||
(is (= ["CREATE INDEX my_column_idx ON my_table (my_column)"]
|
||||
(sql/format {:create-index [:my-column-idx [:my-table :my-column]]})))
|
||||
(is (= ["CREATE INDEX my_column_idx ON my_table (my_column)"]
|
||||
(sql/format (create-index :my-column-idx [:my-table :my-column]))))
|
||||
(is (= ["CREATE UNIQUE INDEX my_column_idx ON my_table (my_column)"]
|
||||
(sql/format (create-index [:unique :my-column-idx] [:my-table :my-column]))))
|
||||
(is (= ["CREATE INDEX my_column_idx ON my_table (my_column, my_other_column)"]
|
||||
(sql/format (create-index :my-column-idx [:my-table :my-column :my-other-column])))))
|
||||
(testing "PostgreSQL extensions (IF NOT EXISTS and expressions)"
|
||||
(is (= ["CREATE INDEX IF NOT EXISTS my_column_idx ON my_table (my_column)"]
|
||||
(sql/format (create-index [:my-column-idx :if-not-exists] [:my-table :my-column]))))
|
||||
(is (= ["CREATE UNIQUE INDEX IF NOT EXISTS my_column_idx ON my_table (my_column)"]
|
||||
(sql/format (create-index [:unique :my-column-idx :if-not-exists] [:my-table :my-column]))))
|
||||
(is (= ["CREATE INDEX my_column_idx ON my_table (LOWER(my_column))"]
|
||||
(sql/format (create-index :my-column-idx [:my-table :%lower.my-column])))))
|
||||
(testing "PostgreSQL extensions (USING GIN/HASH)"
|
||||
(is (= ["CREATE INDEX my_column_idx ON my_table USING GIN (my_column)"]
|
||||
(sql/format {:create-index [:my-column-idx [:my-table :using-gin :my-column]]})))
|
||||
(is (= ["CREATE INDEX my_column_idx ON my_table USING GIN (my_column)"]
|
||||
(sql/format (create-index :my-column-idx [:my-table :using-gin :my-column]))))
|
||||
(is (= ["CREATE INDEX my_column_idx ON my_table USING HASH (my_column)"]
|
||||
(sql/format {:create-index [:my-column-idx [:my-table :using-hash :my-column]]})))
|
||||
(is (= ["CREATE INDEX my_column_idx ON my_table USING HASH (my_column)"]
|
||||
(sql/format (create-index :my-column-idx [:my-table :using-hash :my-column]))))))
|
||||
|
||||
(deftest join-with-alias
|
||||
(is (= ["SELECT * FROM foo LEFT JOIN (populatons AS pm INNER JOIN customers AS pc ON (pm.id = pc.id) AND (pm.other_id = pc.other_id)) ON foo.fk_id = pm.id"]
|
||||
(sql/format {:select :*
|
||||
:from :foo
|
||||
:left-join [[[:join [:populatons :pm]
|
||||
{:join [[:customers :pc]
|
||||
[:and
|
||||
[:= :pm/id :pc/id]
|
||||
[:= :pm/other-id :pc/other-id]]]}]]
|
||||
[:= :foo/fk-id :pm/id]]}))))
|
||||
|
|
|
|||
|
|
@ -1,4 +1,4 @@
|
|||
;; copyright (c) 2022 sean corfield, all rights reserved
|
||||
;; copyright (c) 2022-2024 sean corfield, all rights reserved
|
||||
|
||||
(ns honey.sql.pg-ops-test
|
||||
(:require [clojure.test :refer [deftest is testing]]
|
||||
|
|
|
|||
|
|
@ -410,10 +410,10 @@
|
|||
|
||||
(deftest select-agg-order-by-test
|
||||
(testing "single expression in order by"
|
||||
(is (= ["SELECT ARRAY_AGG(a ORDER BY x) FROM products"])
|
||||
(sql/format
|
||||
{:select [[[:array_agg [:order-by :a :x]]]]
|
||||
:from :products})))
|
||||
(is (= ["SELECT ARRAY_AGG(a ORDER BY x ASC) FROM products"]
|
||||
(sql/format
|
||||
{:select [[[:array_agg [:order-by :a :x]]]]
|
||||
:from :products}))))
|
||||
(testing "multiple expressions in order by"
|
||||
(is (= ["SELECT ARRAY_AGG(a ORDER BY x ASC, y DESC, z ASC) FROM products"]
|
||||
(sql/format
|
||||
|
|
|
|||
148
test/honey/sql/xtdb_test.cljc
Normal file
148
test/honey/sql/xtdb_test.cljc
Normal file
|
|
@ -0,0 +1,148 @@
|
|||
;; copyright (c) 2020-2025 sean corfield, all rights reserved
|
||||
|
||||
(ns honey.sql.xtdb-test
|
||||
(:require [clojure.test :refer [deftest is testing]]
|
||||
[honey.sql :as sql]
|
||||
[honey.sql.helpers :as h
|
||||
:refer [select exclude rename from]]))
|
||||
|
||||
(deftest select-tests
|
||||
(testing "select, exclude, rename"
|
||||
(is (= ["SELECT * EXCLUDE _id RENAME value AS foo_value FROM foo"]
|
||||
(sql/format (-> (select :*) (exclude :_id) (rename [:value :foo_value])
|
||||
(from :foo)))))
|
||||
(is (= ["SELECT * EXCLUDE (_id, a) RENAME value AS foo_value FROM foo"]
|
||||
(sql/format (-> (select :*) (exclude :_id :a) (rename [:value :foo_value])
|
||||
(from :foo)))))
|
||||
(is (= ["SELECT * EXCLUDE _id RENAME (value AS foo_value, a AS b) FROM foo"]
|
||||
(sql/format (-> (select :*) (exclude :_id)
|
||||
(rename [:value :foo_value]
|
||||
[:a :b])
|
||||
(from :foo)))))
|
||||
(is (= ["SELECT * EXCLUDE _id RENAME value AS foo_value, c.x FROM foo"]
|
||||
(sql/format (-> (select [:* (-> (exclude :_id) (rename [:value :foo_value]))]
|
||||
:c.x)
|
||||
(from :foo)))))
|
||||
(is (= ["SELECT * EXCLUDE (_id, a) RENAME value AS foo_value, c.x FROM foo"]
|
||||
(sql/format (-> (select [:* (-> (exclude :_id :a) (rename [:value :foo_value]))]
|
||||
:c.x)
|
||||
(from :foo)))))
|
||||
(is (= ["SELECT * EXCLUDE _id RENAME (value AS foo_value, a AS b), c.x FROM foo"]
|
||||
(sql/format (-> (select [:* (-> (exclude :_id)
|
||||
(rename [:value :foo_value]
|
||||
[:a :b]))]
|
||||
:c.x)
|
||||
(from :foo))))))
|
||||
(testing "select, nest_one, nest_many"
|
||||
(is (= ["SELECT a._id, NEST_ONE (SELECT * FROM foo AS b WHERE b_id = a._id) FROM bar AS a"]
|
||||
(sql/format '{select (a._id,
|
||||
((nest_one {select * from ((foo b)) where (= b_id a._id)})))
|
||||
from ((bar a))})))
|
||||
(is (= ["SELECT a._id, NEST_MANY (SELECT * FROM foo AS b) FROM bar AS a"]
|
||||
(sql/format '{select (a._id,
|
||||
((nest_many {select * from ((foo b))})))
|
||||
from ((bar a))})))))
|
||||
|
||||
(deftest dotted-array-access-tests
|
||||
(is (= ["SELECT (a.b).c"] ; old, partial support:
|
||||
(sql/format '{select (((. (nest :a.b) :c)))})))
|
||||
(is (= ["SELECT (a.b).c"] ; new, complete support:
|
||||
(sql/format '{select (((:get-in :a.b :c)))})))
|
||||
(is (= ["SELECT (a).b.c"] ; the first expression is always parenthesized:
|
||||
(sql/format '{select (((:get-in :a :b :c)))}))))
|
||||
|
||||
(deftest erase-from-test
|
||||
(is (= ["ERASE FROM foo WHERE foo.id = ?" 42]
|
||||
(-> {:erase-from :foo
|
||||
:where [:= :foo.id 42]}
|
||||
(sql/format))))
|
||||
(is (= ["ERASE FROM foo WHERE foo.id = ?" 42]
|
||||
(-> (h/erase-from :foo)
|
||||
(h/where [:= :foo.id 42])
|
||||
(sql/format)))))
|
||||
|
||||
(deftest inline-record-body
|
||||
(is (= ["{_id: 1, name: 'foo', info: {contact: [{loc: 'home', tel: '123'}, {loc: 'work', tel: '456'}]}}"]
|
||||
(sql/format [:inline {:_id 1 :name "foo"
|
||||
:info {:contact [{:loc "home" :tel "123"}
|
||||
{:loc "work" :tel "456"}]}}]))))
|
||||
|
||||
(deftest records-statement
|
||||
(testing "auto-lift maps"
|
||||
(is (= ["RECORDS ?, ?" {:_id 1 :name "cat"} {:_id 2 :name "dog"}]
|
||||
(sql/format {:records [{:_id 1 :name "cat"}
|
||||
{:_id 2 :name "dog"}]}))))
|
||||
(testing "explicit inline"
|
||||
(is (= ["RECORDS {_id: 1, name: 'cat'}, {_id: 2, name: 'dog'}"]
|
||||
(sql/format {:records [[:inline {:_id 1 :name "cat"}]
|
||||
[:inline {:_id 2 :name "dog"}]]}))))
|
||||
(testing "insert with records"
|
||||
(is (= ["INSERT INTO foo RECORDS {_id: 1, name: 'cat'}, {_id: 2, name: 'dog'}"]
|
||||
(sql/format {:insert-into :foo
|
||||
:records [[:inline {:_id 1 :name "cat"}]
|
||||
[:inline {:_id 2 :name "dog"}]]})))
|
||||
(is (= ["INSERT INTO foo RECORDS {_id: 1, name: 'cat'}, {_id: 2, name: 'dog'}"]
|
||||
(sql/format {:insert-into :foo
|
||||
:records [[:inline {:_id 1 :name "cat"}]
|
||||
[:inline {:_id 2 :name "dog"}]]})))
|
||||
(is (= ["INSERT INTO foo RECORDS ?, ?" {:_id 1 :name "cat"} {:_id 2 :name "dog"}]
|
||||
(sql/format {:insert-into [:foo ; as a sub-clause
|
||||
{:records [{:_id 1 :name "cat"}
|
||||
{:_id 2 :name "dog"}]}]})))))
|
||||
|
||||
(deftest patch-statement
|
||||
(testing "patch with records"
|
||||
(is (= ["PATCH INTO foo RECORDS {_id: 1, name: 'cat'}, {_id: 2, name: 'dog'}"]
|
||||
(sql/format {:patch-into [:foo]
|
||||
:records [[:inline {:_id 1 :name "cat"}]
|
||||
[:inline {:_id 2 :name "dog"}]]})))
|
||||
(is (= ["PATCH INTO foo RECORDS ?, ?" {:_id 1 :name "cat"} {:_id 2 :name "dog"}]
|
||||
(sql/format {:patch-into [:foo ; as a sub-clause
|
||||
{:records [{:_id 1 :name "cat"}
|
||||
{:_id 2 :name "dog"}]}]})))
|
||||
(is (= ["PATCH INTO foo RECORDS ?, ?" {:_id 1 :name "cat"} {:_id 2 :name "dog"}]
|
||||
(sql/format (h/patch-into :foo
|
||||
(h/records [{:_id 1 :name "cat"}
|
||||
{:_id 2 :name "dog"}])))))))
|
||||
|
||||
(deftest object-record-expr
|
||||
(testing "object literal"
|
||||
(is (= ["SELECT OBJECT (_id: 1, name: 'foo')"]
|
||||
(sql/format {:select [[[:object {:_id 1 :name "foo"}]]]})))
|
||||
(is (= ["SELECT OBJECT (_id: 1, name: 'foo')"]
|
||||
(sql/format '{select (((:object {:_id 1 :name "foo"})))}))))
|
||||
(testing "record literal"
|
||||
(is (= ["SELECT RECORD (_id: 1, name: 'foo')"]
|
||||
(sql/format {:select [[[:record {:_id 1 :name "foo"}]]]})))
|
||||
(is (= ["SELECT RECORD (_id: 1, name: 'foo')"]
|
||||
(sql/format '{select (((:record {:_id 1 :name "foo"})))}))))
|
||||
(testing "inline map literal"
|
||||
(is (= ["SELECT {_id: 1, name: 'foo'}"]
|
||||
(sql/format {:select [[[:inline {:_id 1 :name "foo"}]]]})))))
|
||||
|
||||
(deftest navigation-dot-index
|
||||
(is (= ["SELECT (a.b).c[1].d"]
|
||||
(sql/format '{select (((get-in a.b c 1 d)))})))
|
||||
(is (= ["SELECT (a.b).c[?].d" 1]
|
||||
(sql/format '{select (((get-in a.b c (lift 1) d)))})))
|
||||
(is (= ["SELECT (a.b).c[?].d" 1]
|
||||
(sql/format '{select (((get-in (. a b) c (lift 1) d)))})))
|
||||
(is (= ["SELECT (OBJECT (_id: 1, b: 'thing').b).c[?].d" 1]
|
||||
(sql/format '{select (((get-in (. (object {_id 1 b "thing"}) b) c (lift 1) d)))}))))
|
||||
|
||||
(deftest assert-statement
|
||||
(testing "quoted sql"
|
||||
(is (= ["ASSERT NOT EXISTS (SELECT 1 FROM users WHERE email = 'james @example.com')"]
|
||||
(sql/format '{assert (not-exists {select 1 from users where (= email "james @example.com")})}
|
||||
:inline true)))
|
||||
(is (= ["ASSERT TRUE"]
|
||||
(sql/format '{assert true}
|
||||
:inline true))))
|
||||
(testing "helper"
|
||||
(is (= ["ASSERT NOT EXISTS (SELECT 1 FROM users WHERE email = 'james @example.com')"]
|
||||
(-> (h/assert [:not-exists {:select 1 :from :users :where [:= :email "james @example.com"]}])
|
||||
(sql/format {:inline true}))))
|
||||
(is (= ["ASSERT NOT EXISTS (SELECT 1 FROM users WHERE email = 'james @example.com')"]
|
||||
(-> {}
|
||||
(h/assert [:not-exists {:select 1 :from :users :where [:= :email "james @example.com"]}])
|
||||
(sql/format {:inline true}))))))
|
||||
|
|
@ -1,4 +1,4 @@
|
|||
;; copyright (c) 2021-2022 sean corfield, all rights reserved
|
||||
;; copyright (c) 2021-2025 sean corfield, all rights reserved
|
||||
|
||||
(ns honey.sql-test
|
||||
(:refer-clojure :exclude [format])
|
||||
|
|
@ -6,7 +6,8 @@
|
|||
[clojure.test :refer [deftest is testing]]
|
||||
[honey.sql :as sut :refer [format]]
|
||||
[honey.sql.helpers :as h])
|
||||
#?(:clj (:import (clojure.lang ExceptionInfo))))
|
||||
#?(:clj (:import (clojure.lang ExceptionInfo)
|
||||
(java.net URLEncoder))))
|
||||
|
||||
(deftest mysql-tests
|
||||
(is (= ["SELECT * FROM `table` WHERE `id` = ?" 1]
|
||||
|
|
@ -21,6 +22,8 @@
|
|||
(sut/format-expr [:is :id nil])))
|
||||
(is (= ["id = TRUE"]
|
||||
(sut/format-expr [:= :id true])))
|
||||
(is (= ["[id] = ?" true]
|
||||
(sut/format [:= :id true] {:dialect :sqlserver})))
|
||||
(is (= ["id IS TRUE"]
|
||||
(sut/format-expr [:is :id true])))
|
||||
(is (= ["id <> TRUE"]
|
||||
|
|
@ -68,6 +71,10 @@
|
|||
(is (= ["INTERVAL ? DAYS" 30]
|
||||
(sut/format-expr [:interval 30 :days]))))
|
||||
|
||||
(deftest issue-486-interval
|
||||
(is (= ["INTERVAL '30 Days'"]
|
||||
(sut/format-expr [:interval "30 Days"]))))
|
||||
|
||||
(deftest issue-455-null
|
||||
(is (= ["WHERE (abc + ?) IS NULL" "abc"]
|
||||
(sut/format {:where [:= [:+ :abc "abc"] nil]}))))
|
||||
|
|
@ -97,6 +104,8 @@
|
|||
(sut/format {:select [:*] :from [:table] :order-by [[[:date :expiry] :desc] :bar]} {:quoted true})))
|
||||
(is (= ["SELECT * FROM \"table\" WHERE DATE_ADD(\"expiry\", INTERVAL ? DAYS) < NOW()" 30]
|
||||
(sut/format {:select [:*] :from [:table] :where [:< [:date_add :expiry [:interval 30 :days]] [:now]]} {:quoted true})))
|
||||
(is (= ["SELECT * FROM \"table\" WHERE DATE_ADD(\"expiry\", INTERVAL '30 Days') < NOW()"]
|
||||
(sut/format {:select [:*] :from [:table] :where [:< [:date_add :expiry [:interval "30 Days"]] [:now]]} {:quoted true})))
|
||||
(is (= ["SELECT * FROM `table` WHERE `id` = ?" 1]
|
||||
(sut/format {:select [:*] :from [:table] :where [:= :id 1]} {:dialect :mysql})))
|
||||
(is (= ["SELECT * FROM \"table\" WHERE \"id\" IN (?, ?, ?, ?)" 1 2 3 4]
|
||||
|
|
@ -127,6 +136,9 @@
|
|||
(is (= ["SELECT * FROM \"table\" WHERE DATE_ADD(\"expiry\", INTERVAL $1 DAYS) < NOW()" 30]
|
||||
(sut/format {:select [:*] :from [:table] :where [:< [:date_add :expiry [:interval 30 :days]] [:now]]}
|
||||
{:quoted true :numbered true})))
|
||||
(is (= ["SELECT * FROM \"table\" WHERE DATE_ADD(\"expiry\", INTERVAL '30 Days') < NOW()"]
|
||||
(sut/format {:select [:*] :from [:table] :where [:< [:date_add :expiry [:interval "30 Days"]] [:now]]}
|
||||
{:quoted true :numbered true})))
|
||||
(is (= ["SELECT * FROM `table` WHERE `id` = $1" 1]
|
||||
(sut/format {:select [:*] :from [:table] :where [:= :id 1]}
|
||||
{:dialect :mysql :numbered true})))
|
||||
|
|
@ -168,8 +180,10 @@
|
|||
["WITH query AS MATERIALIZED (SELECT foo FROM bar)"]))
|
||||
(is (= (format {:with [[:query {:select [:foo] :from [:bar]} :not-materialized]]})
|
||||
["WITH query AS NOT MATERIALIZED (SELECT foo FROM bar)"]))
|
||||
(is (= (format {:with [[:query {:select [:foo] :from [:bar]} :unknown]]})
|
||||
["WITH query AS (SELECT foo FROM bar)"]))
|
||||
(is (= (format {:with [[:query {:select [:foo] :from [:bar]} :kw-1 :kw-2]]})
|
||||
["WITH query AS (SELECT foo FROM bar) KW 1 kw_2"]))
|
||||
(is (= (format {:with-recursive [[:query {:select [:foo] :from [:bar]} :cycle [:a :b :c] :set :d :to [:abs :e] :default 42 :using :x]]})
|
||||
["WITH RECURSIVE query AS (SELECT foo FROM bar) CYCLE a, b, c SET d TO ABS(e) DEFAULT ? USING x" 42]))
|
||||
(is (= (format {:with [[:query1 {:select [:foo] :from [:bar]}]
|
||||
[:query2 {:select [:bar] :from [:quux]}]]
|
||||
:select [:query1.id :query2.name]
|
||||
|
|
@ -207,7 +221,26 @@
|
|||
:from [:hits :stuff]
|
||||
:where [:= :EventDate :ts_upper_bound]})
|
||||
["WITH ? AS ts_upper_bound, stuff AS (SELECT * FROM songs) SELECT * FROM hits, stuff WHERE EventDate = ts_upper_bound"
|
||||
"2019-08-01 15:23:00"]))))
|
||||
"2019-08-01 15:23:00"])))
|
||||
(testing "Use expression in a WITH clause"
|
||||
(is (= (format
|
||||
{:with [[:s [:sum :bytes]]]
|
||||
:select [:s]
|
||||
:from [:table]})
|
||||
["WITH SUM(bytes) AS s SELECT s FROM table"]))
|
||||
|
||||
(is (= (format
|
||||
{:with [[:v [:raw "m['k']"]]]
|
||||
:select [:v]
|
||||
:from [:table]})
|
||||
["WITH m['k'] AS v SELECT v FROM table"]))
|
||||
|
||||
(is (= (format
|
||||
{:with [[:cond [:and [:= :a 1] [:= :b 2] [:= :c 3]]]]
|
||||
:select [:v]
|
||||
:from [:table]
|
||||
:where :cond})
|
||||
["WITH (a = ?) AND (b = ?) AND (c = ?) AS cond SELECT v FROM table WHERE cond" 1 2 3]))))
|
||||
|
||||
(deftest insert-into
|
||||
(is (= (format {:insert-into :foo})
|
||||
|
|
@ -287,7 +320,15 @@
|
|||
(is (= (format {:select [[[:array [] :integer]]]})
|
||||
["SELECT ARRAY[]::INTEGER[]"]))
|
||||
(is (= (format {:select [[[:array [1 2] :text]]]})
|
||||
["SELECT ARRAY[?, ?]::TEXT[]" 1 2]))))
|
||||
["SELECT ARRAY[?, ?]::TEXT[]" 1 2])))
|
||||
(testing "array subquery"
|
||||
(is (= (format {:select [[[:array {:select [:foo] :from [:bar]}]]]})
|
||||
["SELECT ARRAY(SELECT foo FROM bar)"]))
|
||||
(is (= (format {:select [[[:array {:select ^{:as :struct} [:foo :bar] :from [:bar]}]]]})
|
||||
["SELECT ARRAY(SELECT AS STRUCT foo, bar FROM bar)"]))
|
||||
;; documented subquery workaround:
|
||||
(is (= (format {:select [[[:'ARRAY {:select [:foo] :from [:bar]}]]]})
|
||||
["SELECT ARRAY (SELECT foo FROM bar)"]))))
|
||||
|
||||
(deftest union-test
|
||||
;; UNION and INTERSECT subexpressions should not be parenthesized.
|
||||
|
|
@ -339,7 +380,11 @@
|
|||
|
||||
(deftest compare-expressions-test
|
||||
(testing "Sequences should be fns when in value/comparison spots"
|
||||
(is (= ["SELECT foo FROM bar WHERE (col1 MOD ?) = (col2 + ?)" 4 4]
|
||||
(is (= ["SELECT foo FROM bar WHERE (col1 % ?) = (col2 + ?)" 4 4]
|
||||
(format {:select [:foo]
|
||||
:from [:bar]
|
||||
:where [:= [:% :col1 4] [:+ :col2 4]]})))
|
||||
(is (= ["SELECT foo FROM bar WHERE MOD(col1, ?) = (col2 + ?)" 4 4]
|
||||
(format {:select [:foo]
|
||||
:from [:bar]
|
||||
:where [:= [:mod :col1 4] [:+ :col2 4]]}))))
|
||||
|
|
@ -530,13 +575,15 @@
|
|||
(-> {:delete-from :foo
|
||||
:where [:= :foo.id 42]}
|
||||
(format :dialect :mysql :pretty true)))))
|
||||
(when (str/starts-with? #?(:clj (clojure-version)
|
||||
:cljs *clojurescript-version*) "1.11")
|
||||
(testing "format can be called with mixed arguments"
|
||||
(is (= ["\nDELETE FROM `foo`\nWHERE `foo`.`id` = ?\n" 42]
|
||||
(-> {:delete-from :foo
|
||||
:where [:= :foo.id 42]}
|
||||
(format :dialect :mysql {:pretty true})))))))
|
||||
(let [version #?(:cljs *clojurescript-version*
|
||||
:default (clojure-version))]
|
||||
(when (or (str/starts-with? version "1.12")
|
||||
(str/starts-with? version "1.11"))
|
||||
(testing "format can be called with mixed arguments"
|
||||
(is (= ["\nDELETE FROM `foo`\nWHERE `foo`.`id` = ?\n" 42]
|
||||
(-> {:delete-from :foo
|
||||
:where [:= :foo.id 42]}
|
||||
(format :dialect :mysql {:pretty true}))))))))
|
||||
|
||||
(deftest delete-from-test
|
||||
(is (= ["DELETE FROM `foo` WHERE `foo`.`id` = ?" 42]
|
||||
|
|
@ -562,11 +609,17 @@
|
|||
(format)))))
|
||||
|
||||
(deftest truncate-test
|
||||
(is (= ["TRUNCATE `foo`"]
|
||||
(is (= ["TRUNCATE TABLE `foo`"]
|
||||
(-> {:truncate :foo}
|
||||
(format {:dialect :mysql}))))
|
||||
(is (= ["TRUNCATE `foo` CONTINUE IDENTITY"]
|
||||
(is (= ["TRUNCATE TABLE `foo` CONTINUE IDENTITY"]
|
||||
(-> {:truncate [:foo :continue :identity]}
|
||||
(format {:dialect :mysql}))))
|
||||
(is (= ["TRUNCATE TABLE `t1`, `t2`"]
|
||||
(-> {:truncate [[:t1 :t2]]}
|
||||
(format {:dialect :mysql}))))
|
||||
(is (= ["TRUNCATE TABLE `t1`, `t2` CONTINUE IDENTITY"]
|
||||
(-> {:truncate [[:t1 :t2] :continue :identity]}
|
||||
(format {:dialect :mysql})))))
|
||||
|
||||
(deftest inlined-values-are-stringified-correctly
|
||||
|
|
@ -678,8 +731,7 @@ VALUES (?, ?, ?, ?, ?, ?)
|
|||
:values [["UA502" "Bananas" 105 "1971-07-13" "Comedy" "82 minutes"]]}
|
||||
{:pretty true})))
|
||||
(is (= ["
|
||||
INSERT INTO films
|
||||
(code, title, did, date_prod, kind)
|
||||
INSERT INTO films (code, title, did, date_prod, kind)
|
||||
VALUES (?, ?, ?, ?, ?)
|
||||
" "T_601", "Yojimo", 106, "1961-06-16", "Drama"]
|
||||
(format {:insert-into :films
|
||||
|
|
@ -694,8 +746,7 @@ VALUES (?, ?, ?, DEFAULT, ?, ?)
|
|||
:values [["UA502" "Bananas" 105 [:default] "Comedy" "82 minutes"]]}
|
||||
{:pretty true})))
|
||||
(is (= ["
|
||||
INSERT INTO films
|
||||
(code, title, did, date_prod, kind)
|
||||
INSERT INTO films (code, title, did, date_prod, kind)
|
||||
VALUES (?, ?, ?, DEFAULT, ?)
|
||||
" "T_601", "Yojimo", 106, "Drama"]
|
||||
(format {:insert-into :films
|
||||
|
|
@ -706,8 +757,7 @@ VALUES (?, ?, ?, DEFAULT, ?)
|
|||
(deftest on-conflict-tests
|
||||
;; these examples are taken from https://www.postgresqltutorial.com/postgresql-upsert/
|
||||
(is (= ["
|
||||
INSERT INTO customers
|
||||
(name, email)
|
||||
INSERT INTO customers (name, email)
|
||||
VALUES ('Microsoft', 'hotline@microsoft.com')
|
||||
ON CONFLICT ON CONSTRAINT customers_name_key
|
||||
DO NOTHING
|
||||
|
|
@ -719,8 +769,7 @@ DO NOTHING
|
|||
:do-nothing true}
|
||||
{:pretty true})))
|
||||
(is (= ["
|
||||
INSERT INTO customers
|
||||
(name, email)
|
||||
INSERT INTO customers (name, email)
|
||||
VALUES ('Microsoft', 'hotline@microsoft.com')
|
||||
ON CONFLICT
|
||||
ON CONSTRAINT customers_name_key
|
||||
|
|
@ -734,8 +783,7 @@ DO NOTHING
|
|||
:do-nothing true}
|
||||
{:pretty true})))
|
||||
(is (= ["
|
||||
INSERT INTO customers
|
||||
(name, email)
|
||||
INSERT INTO customers (name, email)
|
||||
VALUES ('Microsoft', 'hotline@microsoft.com')
|
||||
ON CONFLICT (name)
|
||||
DO NOTHING
|
||||
|
|
@ -747,8 +795,7 @@ DO NOTHING
|
|||
:do-nothing true}
|
||||
{:pretty true})))
|
||||
(is (= ["
|
||||
INSERT INTO customers
|
||||
(name, email)
|
||||
INSERT INTO customers (name, email)
|
||||
VALUES ('Microsoft', 'hotline@microsoft.com')
|
||||
ON CONFLICT (name)
|
||||
DO NOTHING
|
||||
|
|
@ -760,21 +807,19 @@ DO NOTHING
|
|||
:do-nothing true}
|
||||
{:pretty true})))
|
||||
(is (= ["
|
||||
INSERT INTO customers
|
||||
(name, email)
|
||||
INSERT INTO customers (name, email)
|
||||
VALUES ('Microsoft', 'hotline@microsoft.com')
|
||||
ON CONFLICT (name, email)
|
||||
ON CONFLICT ((foo + ?), name, (TRIM(email)))
|
||||
DO NOTHING
|
||||
"]
|
||||
" 1]
|
||||
(format {:insert-into :customers
|
||||
:columns [:name :email]
|
||||
:values [[[:inline "Microsoft"], [:inline "hotline@microsoft.com"]]]
|
||||
:on-conflict [:name :email]
|
||||
:on-conflict [[:+ :foo 1] :name [:trim :email]]
|
||||
:do-nothing true}
|
||||
{:pretty true})))
|
||||
(is (= ["
|
||||
INSERT INTO customers
|
||||
(name, email)
|
||||
INSERT INTO customers (name, email)
|
||||
VALUES ('Microsoft', 'hotline@microsoft.com')
|
||||
ON CONFLICT (name)
|
||||
DO UPDATE SET email = EXCLUDED.email || ';' || customers.email
|
||||
|
|
@ -848,7 +893,7 @@ ORDER BY id = ? DESC
|
|||
:order-by [(keyword sort-column)]}
|
||||
(format))
|
||||
(is false "; not detected in entity!")
|
||||
(catch #?(:clj Throwable :cljs :default) e
|
||||
(catch #?(:cljs :default :default Exception) e
|
||||
(is (:disallowed (ex-data e))))))))
|
||||
;; should not produce: ["SELECT foo, bar FROM mytable ORDER BY foo; select * from users"]
|
||||
|
||||
|
|
@ -1045,6 +1090,12 @@ ORDER BY id = ? DESC
|
|||
(is (= ["SELECT `A``B`"] (sut/format {:select (keyword "A`B")} {:dialect :mysql})))
|
||||
(is (= ["SELECT \"A\"\"B\""] (sut/format {:select (keyword "A\"B")} {:dialect :oracle}))))
|
||||
|
||||
(deftest issue-407-temporal
|
||||
(is (= ["SELECT f.* FROM foo FOR SYSTEM_TIME ALL AS f WHERE f.id = ?" 1]
|
||||
(sut/format {:select :f.* :from [[:foo :f :for :system-time :all]] :where [:= :f.id 1]})))
|
||||
(is (= ["SELECT * FROM foo FOR SYSTEM_TIME ALL WHERE id = ?" 1]
|
||||
(sut/format {:select :* :from [[:foo :for :system-time :all]] :where [:= :id 1]}))))
|
||||
|
||||
(deftest issue-421-mysql-replace-into
|
||||
(is (= ["INSERT INTO table VALUES (?, ?, ?)" 1 2 3]
|
||||
(sut/format {:insert-into :table :values [[1 2 3]]})))
|
||||
|
|
@ -1133,9 +1184,10 @@ ORDER BY id = ? DESC
|
|||
|
||||
(deftest issue-474-dot-selection
|
||||
(testing "basic dot selection"
|
||||
(is (= ["SELECT a.b, c.d, a.d.x"]
|
||||
(is (= ["SELECT a.b, c.d, a.d.x, a.d.x.y"]
|
||||
(let [t :a c :d]
|
||||
(sut/format {:select [[[:. t :b]] [[:. :c c]] [[:. t c :x]]]}))))
|
||||
(sut/format {:select [[[:. t :b]] [[:. :c c]]
|
||||
[[:. t c :x]] [[:. t c :x :y]]]}))))
|
||||
(is (= ["SELECT [a].[b], [c].[d], [a].[d].[x]"]
|
||||
(let [t :a c :d]
|
||||
(sut/format {:select [[[:. t :b]] [[:. :c c]] [[:. t c :x]]]}
|
||||
|
|
@ -1149,8 +1201,54 @@ ORDER BY id = ? DESC
|
|||
(sut/format '{select (((. (nest v) *))
|
||||
((. (nest w) x))
|
||||
((. (nest (y z)) *)))}
|
||||
{:dialect :mysql})))
|
||||
(is (= ["SELECT (v).*, (w).x, (Y(z)).*"]
|
||||
(sut/format '{select (((get-in v *))
|
||||
((get-in w x))
|
||||
((get-in (y z) *)))})))
|
||||
(is (= ["SELECT (`v`).*, (`w`).`x`, (Y(`z`)).*"]
|
||||
(sut/format '{select (((get-in v *))
|
||||
((get-in w x))
|
||||
((get-in (y z) *)))}
|
||||
{:dialect :mysql})))))
|
||||
|
||||
(deftest issue-570-snowflake-dot-selection
|
||||
(testing "basic colon selection"
|
||||
(is (= ["SELECT a:b, c:d, a:d.x, a:d.x.y"]
|
||||
(let [t :a c :d]
|
||||
(sut/format {:select [[[:.:. t :b]] [[:.:. :c c]]
|
||||
[[:.:. t c :x]] [[:.:. t c :x :y]]]}))))
|
||||
(is (= ["SELECT [a]:[b], [c]:[d], [a]:[d].[x]"]
|
||||
(let [t :a c :d]
|
||||
(sut/format {:select [[[:.:. t :b]] [[:.:. :c c]] [[:.:. t c :x]]]}
|
||||
{:dialect :sqlserver})))))
|
||||
(testing "basic field selection from composite"
|
||||
(is (= ["SELECT (v):*, (w):x, (Y(z)):*"]
|
||||
(sut/format '{select (((.:. (nest v) *))
|
||||
((.:. (nest w) x))
|
||||
((.:. (nest (y z)) *)))})))
|
||||
(is (= ["SELECT (`v`):*, (`w`):`x`, (Y(`z`)):*"]
|
||||
(sut/format '{select (((.:. (nest v) *))
|
||||
((.:. (nest w) x))
|
||||
((.:. (nest (y z)) *)))}
|
||||
{:dialect :mysql}))))
|
||||
(testing "bracket selection"
|
||||
(is (= ["SELECT a['b'], c['b'], a['d'].x, a:e[0].name"]
|
||||
(sut/format {:select [[[:at :a [:inline "b"]]]
|
||||
[[:at :c "b"]]
|
||||
[[:at :a [:inline "d"] :x]]
|
||||
[[:.:. :a [:at :e [:inline 0]] :name]]]})))
|
||||
(is (= ["SELECT a[?].name" 0]
|
||||
(sut/format '{select (((at a (lift 0) name)))})))
|
||||
;; sanity check, compare with get-in:
|
||||
(is (= ["SELECT (a)[?].name" 0]
|
||||
(sut/format '{select (((get-in a (lift 0) name)))})))
|
||||
(is (= ["SELECT (a)['b'], (c)['b'], (a)['d'].x, a:(e)[0].name"]
|
||||
(sut/format {:select [[[:get-in :a [:inline "b"]]]
|
||||
[[:get-in :c "b"]]
|
||||
[[:get-in :a [:inline "d"] :x]]
|
||||
[[:.:. :a [:get-in :e [:inline 0]] :name]]]})))))
|
||||
|
||||
(deftest issue-476-raw
|
||||
(testing "single argument :raw"
|
||||
(is (= ["@foo := 42"]
|
||||
|
|
@ -1170,3 +1268,263 @@ ORDER BY id = ? DESC
|
|||
(sut/format [:raw "@foo := " [42]])))
|
||||
(is (= ["@foo := MYFUNC(?)" 42]
|
||||
(sut/format [:raw "@foo := " [:myfunc 42]])))))
|
||||
|
||||
(deftest issue-483-join
|
||||
(testing "single nested join"
|
||||
(is (= ["SELECT * FROM tbl1 LEFT JOIN (tbl2 INNER JOIN tbl3 USING (common_column)) ON (tbl2.col2 = tbl1.col2) AND (tbl3.col3 = tbl1.col3)"]
|
||||
(-> {:select :*
|
||||
:from :tbl1
|
||||
:left-join [[[:join :tbl2 {:join [:tbl3 [:using [:common_column]]]}]]
|
||||
[:and
|
||||
[:= :tbl2.col2 :tbl1.col2]
|
||||
[:= :tbl3.col3 :tbl1.col3]]]}
|
||||
(sut/format)))))
|
||||
(testing "multiple nested join"
|
||||
(is (= ["SELECT * FROM tbl1 LEFT JOIN (tbl2 INNER JOIN tbl3 USING (common_column) RIGHT JOIN tbl4 USING (id)) ON (tbl2.col2 = tbl1.col2) AND (tbl3.col3 = tbl1.col3)"]
|
||||
(-> {:select :*
|
||||
:from :tbl1
|
||||
:left-join [[[:join :tbl2
|
||||
{:join [:tbl3 [:using [:common_column]]]}
|
||||
{:right-join [:tbl4 [:using :id]]}]]
|
||||
[:and
|
||||
[:= :tbl2.col2 :tbl1.col2]
|
||||
[:= :tbl3.col3 :tbl1.col3]]]}
|
||||
(sut/format)))))
|
||||
(testing "special syntax example"
|
||||
(is (= ["INNER JOIN (tbl1 LEFT JOIN tbl2 USING (id))"]
|
||||
(sut/format {:join [[[:join :tbl1 {:left-join [:tbl2 [:using :id]]}]]]})))))
|
||||
|
||||
#?(:clj
|
||||
(deftest issue-495-formatv
|
||||
(is (= ["SELECT * FROM foo WHERE x = ?" 13]
|
||||
(let [v 13 x 42]
|
||||
(assert x) ; just to mark it as used
|
||||
(sut/formatv [v] '{select * from foo where (= x v)}))))))
|
||||
|
||||
(deftest issue-496-overriding
|
||||
(is (= ["INSERT INTO table (a, b) OVERRIDING SYSTEM VALUE VALUES (?, ?)" 1 2]
|
||||
(sut/format {:insert-into [{:overriding-value :system} :table]
|
||||
:columns [:a :b]
|
||||
:values [[1 2]]})))
|
||||
(is (= ["INSERT INTO table (a, b) OVERRIDING USER VALUE VALUES (?, ?)" 1 2]
|
||||
(sut/format {:insert-into [{:overriding-value :user} :table [:a :b]]
|
||||
:values [[1 2]]})))
|
||||
(is (= ["INSERT INTO table (a, b) OVERRIDING SYSTEM VALUE VALUES (?, ?)" 1 2]
|
||||
(sut/format {:insert-into [{:overriding-value :system} :table]
|
||||
:values [{:a 1 :b 2}]}))))
|
||||
|
||||
(deftest issue-497-alias
|
||||
|
||||
(is (= ["SELECT column_name AS \"some-alias\" FROM b ORDER BY \"some-alias\" ASC"]
|
||||
(sut/format {:select [[:column-name "some-alias"]]
|
||||
:from :b
|
||||
:order-by [[[:raw "\"some-alias\""]]]})))
|
||||
;; likely illegal SQL, but shows quoted keyword escaping the -/_ replace:
|
||||
(is (= ["SELECT column_name AS \"some-alias\" FROM b ORDER BY some-alias ASC"]
|
||||
(sut/format {:select [[:column-name "some-alias"]]
|
||||
:from :b
|
||||
:order-by [[:'some-alias]]})))
|
||||
(is (= ["SELECT column_name AS \"some-alias\" FROM b ORDER BY \"some-alias\" ASC"]
|
||||
(sut/format {:select [[:column-name "some-alias"]]
|
||||
:from :b
|
||||
:order-by [[[:alias "some-alias"]]]})))
|
||||
(is (= ["SELECT column_name AS \"some-alias\" FROM b ORDER BY some_alias ASC"]
|
||||
(sut/format {:select [[:column-name "some-alias"]]
|
||||
:from :b
|
||||
:order-by [[[:alias :some-alias]]]})))
|
||||
(is (= ["SELECT column_name AS \"some-alias\" FROM b ORDER BY \"some-alias\" ASC"]
|
||||
(sut/format {:select [[:column-name "some-alias"]]
|
||||
:from :b
|
||||
:order-by [[[:alias :'some-alias]]]})))
|
||||
(is (= ["SELECT column_name AS \"some-alias\" FROM b ORDER BY \"some-alias\" ASC"]
|
||||
(sut/format {:select [[:column-name "some-alias"]]
|
||||
:from :b
|
||||
:order-by [[[:alias "some-alias"]]]})))
|
||||
(is (= ["SELECT \"column-name\" AS \"some-alias\" FROM \"b\" ORDER BY ? ASC"
|
||||
"some-alias"]
|
||||
(sut/format {:select [[:column-name "some-alias"]]
|
||||
:from :b
|
||||
:order-by ["some-alias"]}
|
||||
{:quoted true})))
|
||||
(is (= ["SELECT `column-name` AS `some-alias` FROM `b` ORDER BY `some-alias` ASC"]
|
||||
(sut/format {:select [[:column-name "some-alias"]]
|
||||
:from :b
|
||||
:order-by [[[:alias "some-alias"]]]}
|
||||
{:dialect :mysql}))))
|
||||
|
||||
(deftest output-clause-post-501
|
||||
(sut/register-clause! :output :select :values)
|
||||
(is (= ["INSERT INTO foo (bar) OUTPUT inserted.* VALUES (?)" 1]
|
||||
(sut/format {:insert-into :foo :output [:inserted.*] :values [{:bar 1}]})))
|
||||
(is (= ["INSERT INTO foo (bar) OUTPUT inserted.* VALUES (?)" 1]
|
||||
(sut/format {:insert-into :foo :columns [:bar] :output [:inserted.*] :values [[1]]}))))
|
||||
|
||||
(deftest at-time-zone-503
|
||||
(is (= ["SELECT foo AT TIME ZONE 'UTC'"]
|
||||
(sut/format {:select [[[:at-time-zone :foo "UTC"]]]})))
|
||||
(is (= ["SELECT foo AT TIME ZONE 'UTC'"]
|
||||
(sut/format {:select [[[:at-time-zone :foo :UTC]]]})))
|
||||
(is (= ["SELECT FOO(bar) AT TIME ZONE 'UTC'"]
|
||||
(sut/format {:select [[[:at-time-zone [:foo :bar] :UTC]]]}))))
|
||||
|
||||
(deftest issue-512
|
||||
(testing "select with metadata"
|
||||
(is (= ["SELECT DISTINCT * FROM table"]
|
||||
(sut/format {:select-distinct [:*] :from [:table]})))
|
||||
(is (= ["SELECT DISTINCT * FROM table"]
|
||||
(sut/format {:select ^{:distinct true} [:*] :from [:table]})))
|
||||
(is (= ["SELECT DISTINCT * FROM table"]
|
||||
(sut/format {:select ^:distinct [:*] :from [:table]})))))
|
||||
|
||||
(deftest issue-515
|
||||
(testing ":always-quoting option"
|
||||
(is (= ["SELECT foo FROM table"]
|
||||
(sut/format '{select foo from table})))
|
||||
(is (= ["SELECT \"foo\" FROM \"table\""]
|
||||
(sut/format '{select foo from table}
|
||||
{:quoted-always #"^(foo|table)$"})))
|
||||
(is (= ["SELECT \"foo\" FROM \"table\""]
|
||||
(sut/format '{select foo from table}
|
||||
{:quoted-always #"^(foo|table)$"
|
||||
:quoted false})))
|
||||
(is (= ["SELECT \"foo\" FROM table"]
|
||||
(sut/format '{select foo from table}
|
||||
{:quoted-always #"^(foo)$"
|
||||
:quoted false})))))
|
||||
|
||||
(deftest issue-520
|
||||
(testing ":inline with a single argument"
|
||||
(is (= ["SELECT 42 AS x"]
|
||||
(sut/format '{select [[[inline 42] x]]}))))
|
||||
(testing ":inline with multiple arguments"
|
||||
(is (= ["SELECT DATE '2024-01-06' AS x"]
|
||||
(sut/format '{select [[[inline DATE "2024-01-06"] x]]}))))
|
||||
(testing ":inline with a parameter"
|
||||
(is (= ["SELECT 42 AS x"]
|
||||
(sut/format '{select [[[inline [param foo]] x]]}
|
||||
{:params {'foo 42}}))))
|
||||
(testing ":inline with a sequence"
|
||||
(is (= ["SELECT ('a', 'b', 'c') AS x"]
|
||||
(sut/format '{select [[[inline ["a" "b" "c"]] x]]}))))
|
||||
(testing ":inline with a lifted sequence"
|
||||
(is (= ["SELECT ['a', 'b', 'c'] AS x"]
|
||||
(sut/format '{select [[[inline [lift ["a" "b" "c"]]] x]]})))))
|
||||
|
||||
(deftest issue-522
|
||||
(testing "from with metadata"
|
||||
(is (= ["SELECT * FROM table WITH (HINT)"]
|
||||
(sut/format {:select [:*] :from [^:hint [:table]]})))
|
||||
;; hash map (metadata) is unordered:
|
||||
(is (or (= ["SELECT * FROM table WITH (ABC, DEF)"]
|
||||
(sut/format {:select [:*] :from [^:abc ^:def [:table]]}))
|
||||
(= ["SELECT * FROM table WITH (DEF, ABC)"]
|
||||
(sut/format {:select [:*] :from [^:abc ^:def [:table]]}))))
|
||||
(is (or (= ["SELECT * FROM table WITH (ABC, DEF)"]
|
||||
(sut/format {:select [:*] :from [^{:abc true :def true} [:table]]}))
|
||||
(= ["SELECT * FROM table WITH (DEF, ABC)"]
|
||||
(sut/format {:select [:*] :from [^{:abc true :def true} [:table]]}))))))
|
||||
|
||||
(deftest issue-527-composite
|
||||
(is (= ["SELECT (a, b) AS c FROM table"]
|
||||
(sut/format {:select [[[:composite :a :b] :c]] :from [:table]})))
|
||||
(is (= ["SELECT a FROM table WHERE (b, c) = (?, ?)" 1 2]
|
||||
(sut/format {:select :a :from :table :where [:= [:composite :b :c] [:composite 1 2]]})))
|
||||
(is (= ["SELECT a, b, c FROM (VALUES (?, ?, ?), (?, ?, ?)) AS t (a, b, c)" 1 2 3 4 5 6]
|
||||
(sut/format {:select [:a :b :c]
|
||||
:from [[{:values [[1 2 3] [4 5 6]]}
|
||||
[:t [:composite :a :b :c]]]]}))))
|
||||
|
||||
(deftest issue-543-param
|
||||
(testing "quoted param with symbol param"
|
||||
(is (= ["SELECT a FROM table WHERE x = ?" 42]
|
||||
(sut/format '{select a from table where (= x (param y))}
|
||||
{:params {'y 42}})))
|
||||
(is (= ["SELECT a FROM table WHERE x = ?" 42]
|
||||
(sut/format '{select a from table where (= x ?y)}
|
||||
{:params {'y 42}}))))
|
||||
(testing "quoted param with keyword param"
|
||||
(is (= ["SELECT a FROM table WHERE x = ?" 42]
|
||||
(sut/format '{select a from table where (= x (param y))}
|
||||
{:params {:y 42}})))
|
||||
(is (= ["SELECT a FROM table WHERE x = ?" 42]
|
||||
(sut/format '{select a from table where (= x :?y)}
|
||||
{:params {:y 42}}))))
|
||||
(testing "all combinations"
|
||||
(doseq [p1 [:y 'y] p2 [:y 'y]]
|
||||
(is (= ["SELECT a FROM table WHERE x = ?" 42]
|
||||
(sut/format {:select :a :from :table :where [:= :x [:param p1]]}
|
||||
{:params {p2 42}}))))))
|
||||
|
||||
(deftest issue-n-using
|
||||
(testing "all keywords"
|
||||
(is (= ["SELECT * FROM `t1` INNER JOIN `t2` USING (`id`) WHERE `t1`.`id` = ?" 1]
|
||||
(sut/format {:select :* :from :t1 :join [:t2 [:using :id]] :where [:= :t1/id 1]} {:dialect :mysql}))))
|
||||
(testing "all symbols"
|
||||
(is (= ["SELECT * FROM `t1` INNER JOIN `t2` USING (`id`) WHERE `t1`.`id` = ?" 1]
|
||||
(sut/format '{select * from t1 join (t2 (using id)) where (= t1/id 1)} {:dialect :mysql}))))
|
||||
(testing "mixed keywords and symbols"
|
||||
(is (= ["SELECT * FROM `t1` INNER JOIN `t2` USING (`id`) WHERE `t1`.`id` = ?" 1]
|
||||
(sut/format '{select * from t1 join (t2 (:using id)) where (= t1/id 1)} {:dialect :mysql})))))
|
||||
|
||||
(deftest issue-548-format-var-encoding
|
||||
(is (= ["CREATE TABLE \"With%20Space\""]
|
||||
(sut/format {:create-table "With%20Space"})))
|
||||
(is (= ["CREATE TABLE \"%20WithLeadingSpace\""]
|
||||
(sut/format {:create-table "%20WithLeadingSpace"})))
|
||||
#?(:clj (let [table (URLEncoder/encode "привіт")]
|
||||
(is (= [(str "CREATE TABLE \"" table "\"")]
|
||||
(sut/format {:create-table table}))))))
|
||||
|
||||
(deftest issue-555-setting
|
||||
(testing "setting default time"
|
||||
(is (= ["SETTING DEFAULT SYSTEM_TIME AS OF DATE '2024-11-24'"]
|
||||
(sut/format {:setting [:default :system-time :as-of [:inline :DATE "2024-11-24"]]})))
|
||||
(is (= ["SETTING SNAPSHOT_TIME TO DATE '2024-11-24', DEFAULT VALID_TIME TO BETWEEN DATE '2022' AND DATE '2023'"]
|
||||
(sut/format {:setting [[:snapshot-time :to [:inline :DATE "2024-11-24"]]
|
||||
[:default :valid-time :to :between [:inline :DATE "2022"] :and [:inline :DATE "2023"]]]})))
|
||||
(is (= ["SETTING DEFAULT SYSTEM_TIME AS OF DATE '2024-11-24' SELECT * FROM table"]
|
||||
(sut/format (-> (h/setting :default :system-time :as-of [:inline :DATE "2024-11-24"])
|
||||
(h/select :*)
|
||||
(h/from :table)))))
|
||||
(is (= ["SETTING SNAPSHOT_TIME TO DATE '2024-11-24', DEFAULT VALID_TIME TO BETWEEN DATE '2022' AND DATE '2023' SELECT * FROM table"]
|
||||
(sut/format (-> (h/setting [:snapshot-time :to [:inline :DATE "2024-11-24"]]
|
||||
[:default :valid-time :to :between [:inline :DATE "2022"] :and [:inline :DATE "2023"]])
|
||||
(h/select :*)
|
||||
(h/from :table)))))))
|
||||
|
||||
(deftest issue-571
|
||||
(testing "an empty where clause is omitted"
|
||||
(is (= ["SELECT * FROM foo"]
|
||||
(sut/format {:select :* :from :foo :where []})))
|
||||
#?(:clj
|
||||
(is (thrown? clojure.lang.ExceptionInfo
|
||||
(sut/format {:select :* :from :foo :where nil}))))
|
||||
(is (= ["SELECT * FROM foo WHERE 1 = 1"]
|
||||
(sut/format {:select :* :from :foo :where [:= 1 1]} {:inline true}))))
|
||||
(testing "an empty order by clause is omitted"
|
||||
(is (= ["SELECT * FROM foo"]
|
||||
(sut/format {:select :* :from :foo :order-by []})))
|
||||
#?(:clj
|
||||
(is (thrown? clojure.lang.ExceptionInfo
|
||||
(sut/format {:select :* :from :foo :order-by nil}))))
|
||||
(is (= ["SELECT * FROM foo ORDER BY bar ASC"]
|
||||
(sut/format {:select :* :from :foo :order-by [:bar]})))))
|
||||
|
||||
(comment
|
||||
;; partial (incorrect!) workaround for #407:
|
||||
(sut/format {:select :f.* :from [[:foo [:f :for :system-time]]] :where [:= :f.id 1]})
|
||||
;; correct version:
|
||||
(sut/format {:select :f.* :from [[:foo :f :for :system-time]] :where [:= :f.id 1]})
|
||||
(sut/format {:where [:= :x [:inline :DATE "2019-01-01"]]})
|
||||
;; https://github.com/seancorfield/honeysql/issues/526
|
||||
(->
|
||||
{:create-table-as [:a-b.b-c.c-d]
|
||||
:select [:*]
|
||||
:from [:a-b.b-c.c-d]}
|
||||
(sut/format {:dialect :nrql}))
|
||||
(sut/format {:select :a:b.c}) ; quotes a:b
|
||||
(sut/format [:. :a :b :c]) ; a.b.c
|
||||
(sut/format [:. :a :b :c :d]) ; drops d ; a.b.c
|
||||
(sut/format [:.:. :a :b :c]) ; .(a, b, c)
|
||||
(sut/format '(.:. a b c)) ; .(a, b, c)
|
||||
)
|
||||
|
|
|
|||
10
test/honey/unhashable_test.clj
Normal file
10
test/honey/unhashable_test.clj
Normal file
|
|
@ -0,0 +1,10 @@
|
|||
(ns honey.unhashable-test
|
||||
(:require [clojure.test :refer [deftest is]]
|
||||
[honey.sql :as sut]))
|
||||
|
||||
(deftest unhashable-value-509
|
||||
(let [unhashable (reify Object
|
||||
(toString [_] "unhashable")
|
||||
(hashCode [_] (throw (ex-info "Unsupported" {}))))]
|
||||
(is (= ["INSERT INTO table VALUES (?)" unhashable]
|
||||
(sut/format {:insert-into :table :values [[unhashable]]})))))
|
||||
|
|
@ -1,4 +1,4 @@
|
|||
;; copyright (c) 2023 sean corfield, all rights reserved
|
||||
;; copyright (c) 2023-2024 sean corfield, all rights reserved
|
||||
|
||||
(ns honey.union-test
|
||||
(:refer-clojure :exclude [format])
|
||||
|
|
|
|||
62
test/honey/util_test.cljc
Normal file
62
test/honey/util_test.cljc
Normal file
|
|
@ -0,0 +1,62 @@
|
|||
(ns honey.util-test
|
||||
(:refer-clojure :exclude [str])
|
||||
(:require [clojure.test :refer [deftest is are]]
|
||||
[honey.sql.util :as sut]))
|
||||
|
||||
(deftest str-test
|
||||
(are [arg1 result] (= result (sut/str arg1))
|
||||
nil ""
|
||||
1 "1"
|
||||
"foo" "foo"
|
||||
:foo ":foo")
|
||||
(are [arg1 arg2 result] (= result (sut/str arg1 arg2))
|
||||
nil nil ""
|
||||
nil 1 "1"
|
||||
1 nil "1"
|
||||
1 2 "12"
|
||||
:foo "bar" ":foobar")
|
||||
(are [arg1 arg2 arg3 result] (= result (sut/str arg1 arg2 arg3))
|
||||
nil nil nil ""
|
||||
nil 1 nil "1"
|
||||
1 nil nil "1"
|
||||
1 nil 2 "12"
|
||||
:foo "bar" 'baz ":foobarbaz")
|
||||
(are [args result] (= result (apply sut/str args))
|
||||
(range 10) "0123456789"
|
||||
[] ""))
|
||||
|
||||
(deftest join-test
|
||||
(is (= "0123456789" (sut/join "" (range 10))))
|
||||
(is (= "1" (sut/join "" [1])))
|
||||
(is (= "" (sut/join "" [])))
|
||||
(is (= "0, 1, 2, 3, 4, 5, 6, 7, 8, 9" (sut/join ", " (range 10))))
|
||||
(is (= "1" (sut/join ", " [1])))
|
||||
(is (= "" (sut/join ", " [])))
|
||||
|
||||
(is (= "0_0, 1_1, 2_2, 3_3, 4_4, 5_5, 6_6, 7_7, 8_8, 9_9"
|
||||
(sut/join ", " (map #(sut/str % "_" %)) (range 10))))
|
||||
(is (= "1_1"
|
||||
(sut/join ", " (map #(sut/str % "_" %)) [1])))
|
||||
(is (= ""
|
||||
(sut/join ", " (map #(sut/str % "_" %)) [])))
|
||||
|
||||
(is (= "1, 2, 3, 4"
|
||||
(sut/join ", " (remove nil?) [1 nil 2 nil 3 nil nil nil 4])))
|
||||
(is (= "" (sut/join ", " (remove nil?) [nil nil nil nil]))))
|
||||
|
||||
(deftest split-by-separator-test
|
||||
(is (= [""] (sut/split-by-separator "" ".")))
|
||||
(is (= ["" ""] (sut/split-by-separator "." ".")))
|
||||
(is (= ["hello"] (sut/split-by-separator "hello" ".")))
|
||||
(is (= ["h" "e" "l" "l" "o"] (sut/split-by-separator "h.e.l.l.o" ".")))
|
||||
(is (= ["" "h" "e" "" "" "l" "" "l" "o" ""]
|
||||
(sut/split-by-separator ".h.e...l..l.o." "."))))
|
||||
|
||||
(deftest into*-test
|
||||
(is (= [1] (sut/into* [1] nil)))
|
||||
(is (= [1] (sut/into* [1] [])))
|
||||
(is (= [1] (sut/into* [1] nil [] nil [])))
|
||||
(is (= [1 2 3] (sut/into* [1] [2 3])))
|
||||
(is (= [1 2 3 4 5 6] (sut/into* [1] [2 3] [4 5 6])))
|
||||
(is (= [1 2 3 4 5 6 7] (sut/into* [1] [2 3] [4 5 6] [7])))
|
||||
(is (= [1 2 3 4 5 6 7 8 9] (sut/into* [1] [2 3] [4 5 6] [7] [8 9]))))
|
||||
Loading…
Reference in a new issue