6
6
*/
7
7
package net.ponec.scriptKt
8
8
9
- import java.io.Closeable
10
9
import java.sql.*
11
10
import java.time.LocalDate
12
11
import java.util.*
@@ -23,17 +22,15 @@ class SqlExecutorKt {
23
22
24
23
companion object Static {
25
24
private val db = ConnectionProvider .forH2(" user" , " pwd" )
26
- @Throws(Exception ::class )
27
- @JvmStatic
25
+ @JvmStatic @Throws(Exception ::class )
28
26
fun main (args : Array <String >) {
29
27
println (" args: [${args.joinToString()} ]" )
30
28
db.connection().use { SqlExecutorKt ().mainStart(it) }
31
29
}
32
30
}
33
31
34
- @Throws(Exception ::class )
35
32
fun mainStart (dbConnection : Connection ) {
36
- SqlParamBuilder (dbConnection).use { builder ->
33
+ SqlParamBuilderKt (dbConnection).use { builder ->
37
34
38
35
println (" # CREATE TABLE" )
39
36
builder.sql("""
@@ -42,8 +39,8 @@ class SqlExecutorKt {
42
39
, name VARCHAR(256) DEFAULT 'test'
43
40
, code VARCHAR(1)
44
41
, created DATE NOT NULL )
45
- """ .trimIndent()
46
- ) .execute()
42
+ """ .trimIndent())
43
+ .execute()
47
44
48
45
println (" # SINGLE INSERT" )
49
46
builder.sql("""
@@ -74,25 +71,22 @@ class SqlExecutorKt {
74
71
.execute()
75
72
76
73
println (" # SELECT" )
77
- val employees = builder.sql("""
74
+ val employees: List < Employee > = builder.sql("""
78
75
SELECT t.id, t.name, t.created
79
76
FROM employee t
80
77
WHERE t.id < :id
81
78
AND t.code IN (:code)
82
79
ORDER BY t.id
83
- """ .trimIndent()
84
- )
80
+ """ .trimIndent())
85
81
.bind(" id" , 10 )
86
82
.bind(" code" , " T" , " V" )
87
83
.streamMap { Employee (
88
84
it.getInt(" id" ),
89
85
it.getString(" name" ),
90
- it.getObject(" created" , LocalDate ::class .java)
91
- )
92
- }
86
+ it.getObject(" created" , LocalDate ::class .java)) }
93
87
.toList()
94
- System . out .printf (" # PRINT RESULT OF: %s%n " , builder.toStringLine())
95
- employees.stream().forEach { x : Employee ? -> println (x ) }
88
+ println (" # PRINT RESULT OF: ${ builder.toStringLine()} " )
89
+ employees.stream().forEach { employee : Employee -> println (employee ) }
96
90
assertEquals(3 , employees.size)
97
91
assertEquals(1 , employees[0 ].id)
98
92
assertEquals(" test" , employees[0 ].name)
@@ -103,8 +97,7 @@ class SqlExecutorKt {
103
97
WHERE t.id < [10]
104
98
AND t.code IN ([T],[V])
105
99
ORDER BY t.id
106
- """ .trimIndent(), builder.toString()
107
- )
100
+ """ .trimIndent(), builder.toString())
108
101
}
109
102
}
110
103
@@ -139,40 +132,36 @@ class SqlExecutorKt {
139
132
}
140
133
141
134
/* *
142
- * Less than 140 lines long class to simplify work with JDBC.
135
+ * Less than 130 lines long class to simplify work with JDBC.
143
136
* Original source: [GitHub](https://github.com/pponec/DirectoryBookmarks/blob/development/src/main/java/net/ponec/script/SqlExecutor.java)
144
137
* Licence: Apache License, Version 2.0
145
138
* @author Pavel Ponec, https://github.com/pponec
146
139
* @version 1.0.7
147
140
*/
148
- internal class SqlParamBuilder (private val connection : Connection
149
- ) : Closeable {
141
+ internal class SqlParamBuilderKt (private val connection : Connection ) : AutoCloseable {
150
142
private val params: MutableMap <String , Array <out Any ?>> = HashMap ()
151
143
private val sqlParameterMark = Pattern .compile(" :(\\ w+)" )
152
- private var sqlTemplate: String = " "
144
+ var sqlTemplate: String = " " ; private set
153
145
private var preparedStatement: PreparedStatement ? = null
154
146
155
- /* * Close statement (if any) and set a new SQL template */
156
- fun sql (vararg sqlLines : String ): SqlParamBuilder {
147
+ /* * Close statement (if any) and set the new SQL template */
148
+ fun sql (vararg sqlLines : String ): SqlParamBuilderKt {
157
149
close()
158
150
sqlTemplate = if (sqlLines.size == 1 ) sqlLines[0 ] else sqlLines.joinToString(" \n " )
159
151
return this
160
152
}
161
153
162
154
/* * Assign a SQL value(s). In case a reused statement set the same number of parameters items. */
163
- fun bind (key : String , vararg values : Any? ): SqlParamBuilder {
155
+ fun bind (key : String , vararg values : Any? ): SqlParamBuilderKt {
164
156
params[key] = if (values.isNotEmpty()) values else arrayOfNulls(1 )
165
157
return this
166
158
}
167
159
168
- @Throws(IllegalStateException ::class , SQLException ::class )
169
- fun execute (): Int {
170
- return prepareStatement().executeUpdate()
171
- }
160
+ fun execute (): Int =
161
+ prepareStatement().executeUpdate()
172
162
173
163
/* * A ResultSet object is automatically closed when the Statement object that generated it is closed,
174
164
* re-executed, or used to retrieve the next result from a sequence of multiple results. */
175
- @Throws(IllegalStateException ::class )
176
165
private fun executeSelect (): ResultSet {
177
166
return try {
178
167
prepareStatement().executeQuery()
@@ -201,14 +190,11 @@ class SqlExecutorKt {
201
190
}
202
191
203
192
/* * Iterate executed select */
204
- @Throws(IllegalStateException ::class , SQLException ::class )
205
- fun forEach (consumer : (ResultSet ) -> Unit ) {
193
+ fun forEach (consumer : (ResultSet ) -> Unit ) =
206
194
stream().forEach(consumer)
207
- }
208
195
209
- fun <R > streamMap (mapper : (ResultSet ) -> R ): Stream <R > {
210
- return stream().map(mapper)
211
- }
196
+ fun <R > streamMap (mapper : (ResultSet ) -> R ): Stream <R > =
197
+ stream().map(mapper)
212
198
213
199
/* * The method closes a PreparedStatement object with related objects, not the database connection. */
214
200
override fun close () {
@@ -237,31 +223,28 @@ class SqlExecutorKt {
237
223
private fun buildSql (sqlValues : MutableList <Any ?>, toLog : Boolean ): String {
238
224
val result = StringBuilder (256 )
239
225
val matcher = sqlParameterMark.matcher(sqlTemplate)
240
- val missingKeys = HashSet <Any >()
226
+ val missingKeys = mutableSetOf <Any >()
241
227
while (matcher.find()) {
242
228
val key = matcher.group(1 )
243
229
val values = params[key]
244
230
if (values != null ) {
245
231
matcher.appendReplacement(result, " " )
246
232
for (i in values.indices) {
247
233
if (i > 0 ) result.append(' ,' )
248
- result.append(Matcher .quoteReplacement(if (toLog) " [" + values[i] + " ]" else " ?" ))
234
+ result.append(Matcher .quoteReplacement(if (toLog) " [${ values[i]} ]" else " ?" ))
249
235
sqlValues.add(values[i])
250
236
}
251
237
} else {
252
238
matcher.appendReplacement(result, Matcher .quoteReplacement(matcher.group()))
253
239
missingKeys.add(key)
254
240
}
255
241
}
256
- require(! (! toLog && ! missingKeys.isEmpty())) { " Missing value of the keys: $missingKeys " }
242
+ require(toLog || missingKeys.isEmpty()) {
243
+ " Missing value of the keys: [${missingKeys.joinToString (" , " )} ]" }
257
244
matcher.appendTail(result)
258
245
return result.toString()
259
246
}
260
247
261
- fun sqlTemplate (): String {
262
- return sqlTemplate
263
- }
264
-
265
248
override fun toString (): String {
266
249
return buildSql(ArrayList (), true )
267
250
}
0 commit comments