blob: 6da8a528943a5a0a9c95c234ab713b83f29aef75 [file] [log] [blame]
danielk197769e777f2006-06-14 10:38:02 +00001# 2006 July 14
2#
3# The author disclaims copyright to this source code. In place of
4# a legal notice, here is a blessing:
5#
6# May you do good and not evil.
7# May you find forgiveness for yourself and forgive others.
8# May you share freely, never taking more than you give.
9#
10#***********************************************************************
11# This file implements regression tests for SQLite library. The
drh1409be62006-08-23 20:07:20 +000012# focus of this script is extension loading.
danielk197769e777f2006-06-14 10:38:02 +000013#
danielk1977eefa0002009-03-20 09:09:37 +000014# $Id: loadext.test,v 1.17 2009/03/20 09:09:37 danielk1977 Exp $
danielk197769e777f2006-06-14 10:38:02 +000015
16set testdir [file dirname $argv0]
17source $testdir/tester.tcl
18
drh1e9daa62007-04-06 21:42:22 +000019ifcapable !load_ext {
20 finish_test
21 return
22}
23
drh428397c2006-06-17 13:21:32 +000024# The name of the test extension varies by operating system.
25#
mistachkinf1c6bc52012-06-21 15:09:20 +000026if {$::tcl_platform(platform) eq "windows"} {
drh428397c2006-06-17 13:21:32 +000027 set testextension ./testloadext.dll
danielk197769e777f2006-06-14 10:38:02 +000028} else {
drh428397c2006-06-17 13:21:32 +000029 set testextension ./libtestloadext.so
danielk197769e777f2006-06-14 10:38:02 +000030}
drh104d74c2008-10-13 17:09:11 +000031set gcc_shared "-shared -fPIC"
danielk197763703a42008-06-30 15:09:29 +000032if {$::tcl_platform(os) eq "Darwin"} {
33 set gcc_shared -dynamiclib
34}
danielk197769e777f2006-06-14 10:38:02 +000035
danielk197798cab2c2007-09-01 05:57:49 +000036# The error messages tested by this file are operating system dependent
37# (because they are returned by sqlite3OsDlError()). For now, they only
38# work with UNIX (and probably only certain kinds of UNIX).
39#
danielk1977c3f759b2007-09-01 06:19:05 +000040# When a shared-object cannot be opened because it does not exist, the
41# format of the message returned is:
danielk197798cab2c2007-09-01 05:57:49 +000042#
danielk1977c3f759b2007-09-01 06:19:05 +000043# [format $dlerror_nosuchfile <shared-object-name>]
44#
45# When a shared-object cannot be opened because it consists of the 4
46# characters "blah" only, we expect the error message to be:
47#
48# [format $dlerror_notadll <shared-object-name>]
danielk197798cab2c2007-09-01 05:57:49 +000049#
50# When a symbol cannot be found within an open shared-object, the error
51# message should be:
52#
53# [format $dlerror_nosymbol <shared-object-name> <symbol-name>]
54#
danielk1977c3f759b2007-09-01 06:19:05 +000055# The exact error messages are not important. The important bit is
56# that SQLite is correctly copying the message from xDlError().
57#
58set dlerror_nosuchfile \
59 {%s: cannot open shared object file: No such file or directory}
60set dlerror_notadll {%s: file too short}
61set dlerror_nosymbol {%s: undefined symbol: %s}
danielk197798cab2c2007-09-01 05:57:49 +000062
danielk197763703a42008-06-30 15:09:29 +000063if {$::tcl_platform(os) eq "Darwin"} {
drh2d588b82018-11-17 14:59:35 +000064 set dlerror_nosuchfile {dlopen.%s, 10.: .*image.*found.*}
65 set dlerror_notadll {dlopen.%1$s, 10.: .*image.*found.*}
66 set dlerror_nosymbol {dlsym.XXX, %2$s.: symbol not found}
danielk197763703a42008-06-30 15:09:29 +000067}
68
mistachkin1925a2e2014-02-24 21:20:25 +000069if {$::tcl_platform(platform) eq "windows"} {
70 set dlerror_nosuchfile {The specified module could not be found.*}
71 set dlerror_notadll {%%1 is not a valid Win32 application.*}
72 set dlerror_nosymbol {The specified procedure could not be found.*}
73}
74
drh428397c2006-06-17 13:21:32 +000075# Make sure the test extension actually exists. If it does not
76# exist, try to create it. If unable to create it, then skip this
77# test file.
78#
danielk197769e777f2006-06-14 10:38:02 +000079if {![file exists $testextension]} {
drh428397c2006-06-17 13:21:32 +000080 set srcdir [file dir $testdir]/src
81 set testextsrc $srcdir/test_loadext.c
danielk1977eefa0002009-03-20 09:09:37 +000082
83 set cmdline [concat exec gcc $gcc_shared]
dand7d19b72016-08-01 16:50:50 +000084 lappend cmdline -Wall -I$srcdir -I. -I.. -g $testextsrc -o $testextension
danielk1977eefa0002009-03-20 09:09:37 +000085
86 if {[catch $cmdline msg]} {
drh428397c2006-06-17 13:21:32 +000087 puts "Skipping loadext tests: Test extension not built..."
88 puts $msg
89 finish_test
90 return
91 }
danielk197769e777f2006-06-14 10:38:02 +000092}
93
94# Test that loading the extension produces the expected results - adding
95# the half() function to the specified database handle.
96#
97do_test loadext-1.1 {
98 catchsql {
99 SELECT half(1.0);
100 }
101} {1 {no such function: half}}
102do_test loadext-1.2 {
drh41449052006-07-06 17:08:48 +0000103 db enable_load_extension 1
drh428397c2006-06-17 13:21:32 +0000104 sqlite3_load_extension db $testextension testloadext_init
danielk197769e777f2006-06-14 10:38:02 +0000105 catchsql {
106 SELECT half(1.0);
107 }
108} {0 0.5}
109
110# Test that a second database connection (db2) can load the extension also.
111#
112do_test loadext-1.3 {
113 sqlite3 db2 test.db
drhf602a162016-04-21 01:58:21 +0000114 sqlite3_db_config db2 SQLITE_DBCONFIG_ENABLE_LOAD_EXTENSION 1
danielk197769e777f2006-06-14 10:38:02 +0000115 catchsql {
116 SELECT half(1.0);
117 } db2
118} {1 {no such function: half}}
119do_test loadext-1.4 {
drh428397c2006-06-17 13:21:32 +0000120 sqlite3_load_extension db2 $testextension testloadext_init
danielk197769e777f2006-06-14 10:38:02 +0000121 catchsql {
122 SELECT half(1.0);
123 } db2
124} {0 0.5}
125
126# Close the first database connection. Then check that the second database
127# can still use the half() function without a problem.
128#
129do_test loadext-1.5 {
130 db close
131 catchsql {
132 SELECT half(1.0);
133 } db2
134} {0 0.5}
135
136db2 close
137sqlite3 db test.db
drhc2e87a32006-06-27 15:16:14 +0000138sqlite3_enable_load_extension db 1
danielk197769e777f2006-06-14 10:38:02 +0000139
140# Try to load an extension for which the file does not exist.
141#
142do_test loadext-2.1 {
mistachkinfda06be2011-08-02 00:57:34 +0000143 forcedelete ${testextension}xx
danielk197769e777f2006-06-14 10:38:02 +0000144 set rc [catch {
drh428397c2006-06-17 13:21:32 +0000145 sqlite3_load_extension db "${testextension}xx"
danielk197769e777f2006-06-14 10:38:02 +0000146 } msg]
147 list $rc $msg
drhcca17c32013-04-19 12:32:52 +0000148} /[list 1 [format $dlerror_nosuchfile ${testextension}xx.*]]/
danielk197769e777f2006-06-14 10:38:02 +0000149
150# Try to load an extension for which the file is not a shared object
151#
152do_test loadext-2.2 {
drhcca17c32013-04-19 12:32:52 +0000153 set fd [open "./notasharedlib.so" w]
154 puts $fd blah
155 close $fd
156 set fd [open "./notasharedlib.dll" w]
danielk197769e777f2006-06-14 10:38:02 +0000157 puts $fd blah
158 close $fd
159 set rc [catch {
drhcca17c32013-04-19 12:32:52 +0000160 sqlite3_load_extension db "./notasharedlib"
danielk197769e777f2006-06-14 10:38:02 +0000161 } msg]
drhcca17c32013-04-19 12:32:52 +0000162 list $rc $msg
163} /[list 1 [format $dlerror_notadll ./notasharedlib.*]]/
danielk197769e777f2006-06-14 10:38:02 +0000164
165# Try to load an extension for which the file is present but the
166# entry point is not.
167#
168do_test loadext-2.3 {
169 set rc [catch {
170 sqlite3_load_extension db $testextension icecream
171 } msg]
danielk197763703a42008-06-30 15:09:29 +0000172 if {$::tcl_platform(os) eq "Darwin"} {
173 regsub {0x[1234567890abcdefABCDEF]*} $msg XXX msg
174 }
danielk197769e777f2006-06-14 10:38:02 +0000175 list $rc $msg
mistachkin1925a2e2014-02-24 21:20:25 +0000176} /[list 1 [format $dlerror_nosymbol $testextension icecream]]/
danielk197769e777f2006-06-14 10:38:02 +0000177
178# Try to load an extension for which the entry point fails (returns non-zero)
179#
180do_test loadext-2.4 {
181 set rc [catch {
182 sqlite3_load_extension db $testextension testbrokenext_init
183 } msg]
184 list $rc $msg
185} {1 {error during initialization: broken!}}
186
drhfdb83b22006-06-17 14:12:47 +0000187############################################################################
188# Tests for the load_extension() SQL function
189#
190
191db close
192sqlite3 db test.db
drhc2e87a32006-06-27 15:16:14 +0000193sqlite3_enable_load_extension db 1
drhfdb83b22006-06-17 14:12:47 +0000194do_test loadext-3.1 {
195 catchsql {
196 SELECT half(5);
197 }
198} {1 {no such function: half}}
199do_test loadext-3.2 {
danielk197763703a42008-06-30 15:09:29 +0000200 set res [catchsql {
drhfdb83b22006-06-17 14:12:47 +0000201 SELECT load_extension($::testextension)
danielk197763703a42008-06-30 15:09:29 +0000202 }]
203 if {$::tcl_platform(os) eq "Darwin"} {
204 regsub {0x[1234567890abcdefABCDEF]*} $res XXX res
drhfdb83b22006-06-17 14:12:47 +0000205 }
danielk197763703a42008-06-30 15:09:29 +0000206 set res
drhcca17c32013-04-19 12:32:52 +0000207} /[list 1 [format $dlerror_nosymbol $testextension sqlite3_.*_init]]/
drhfdb83b22006-06-17 14:12:47 +0000208do_test loadext-3.3 {
209 catchsql {
210 SELECT load_extension($::testextension,'testloadext_init')
211 }
212} {0 {{}}}
213do_test loadext-3.4 {
214 catchsql {
215 SELECT half(5);
216 }
217} {0 2.5}
drhd72a8412008-06-19 15:44:00 +0000218do_test loadext-3.5 {
219 db eval {
220 SELECT sqlite3_status('MEMORY_USED') AS mused
221 } break
222 puts -nonewline " (memory_used=$mused) "
223 expr {$mused>0}
224} {1}
225do_test loadext-3.6 {
226 catchsql {
227 SELECT sqlite3_status('MEMORY_USED_X') AS mused
228 }
229} {1 {unknown status property: MEMORY_USED_X}}
230do_test loadext-3.7 {
231 catchsql {
232 SELECT sqlite3_status(4.53) AS mused
233 }
234} {1 {unknown status type}}
235do_test loadext-3.8 {
236 catchsql {
237 SELECT sqlite3_status(23) AS mused
238 }
239} {1 {sqlite3_status(23,...) returns 21}}
drhfdb83b22006-06-17 14:12:47 +0000240
drhc2e87a32006-06-27 15:16:14 +0000241# Ticket #1863
242# Make sure the extension loading mechanism will not work unless it
243# is explicitly enabled.
244#
245db close
246sqlite3 db test.db
247do_test loadext-4.1 {
248 catchsql {
249 SELECT load_extension($::testextension,'testloadext_init')
250 }
251} {1 {not authorized}}
252do_test loadext-4.2 {
253 sqlite3_enable_load_extension db 1
254 catchsql {
255 SELECT load_extension($::testextension,'testloadext_init')
256 }
257} {0 {{}}}
258
drhf602a162016-04-21 01:58:21 +0000259# disable all extension loading
drhc2e87a32006-06-27 15:16:14 +0000260do_test loadext-4.3 {
261 sqlite3_enable_load_extension db 0
262 catchsql {
263 SELECT load_extension($::testextension,'testloadext_init')
264 }
265} {1 {not authorized}}
266
drhf602a162016-04-21 01:58:21 +0000267# enable C-api extension loading only. Show that the SQL function
268# still does not work.
269do_test loadext-4.4 {
270 sqlite3_db_config db SQLITE_DBCONFIG_ENABLE_LOAD_EXTENSION 1
271 catchsql {
272 SELECT load_extension($::testextension,'testloadext_init')
273 }
274} {1 {not authorized}}
275
drh701bb3b2008-08-02 03:50:39 +0000276source $testdir/malloc_common.tcl
drhc2e87a32006-06-27 15:16:14 +0000277
278
drh701bb3b2008-08-02 03:50:39 +0000279# Malloc failure in sqlite3_auto_extension and sqlite3_load_extension
280#
281do_malloc_test loadext-5 -tclprep {
282 sqlite3_reset_auto_extension
283} -tclbody {
284 if {[autoinstall_test_functions]==7} {error "out of memory"}
285}
mistachkin1925a2e2014-02-24 21:20:25 +0000286
287# On Windows, this malloc test must be skipped because the winDlOpen
288# function itself can fail due to "out of memory" conditions.
289#
290if {$::tcl_platform(platform) ne "windows"} {
291 do_malloc_test loadext-6 -tclbody {
292 db enable_load_extension 1
293 sqlite3_load_extension db $::testextension testloadext_init
294 }
drh701bb3b2008-08-02 03:50:39 +0000295}
mistachkin1925a2e2014-02-24 21:20:25 +0000296
drh701bb3b2008-08-02 03:50:39 +0000297autoinstall_test_functions
298
danielk197769e777f2006-06-14 10:38:02 +0000299finish_test