tester1.c-pp.js 134 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476
  1. /*
  2. 2022-10-12
  3. The author disclaims copyright to this source code. In place of a
  4. legal notice, here is a blessing:
  5. * May you do good and not evil.
  6. * May you find forgiveness for yourself and forgive others.
  7. * May you share freely, never taking more than you give.
  8. ***********************************************************************
  9. Main functional and regression tests for the sqlite3 WASM API.
  10. This mini-framework works like so:
  11. This script adds a series of test groups, each of which contains an
  12. arbitrary number of tests, into a queue. After loading of the
  13. sqlite3 WASM/JS module is complete, that queue is processed. If any
  14. given test fails, the whole thing fails. This script is built such
  15. that it can run from the main UI thread or worker thread. Test
  16. groups and individual tests can be assigned a predicate function
  17. which determines whether to run them or not, and this is
  18. specifically intended to be used to toggle certain tests on or off
  19. for the main/worker threads or the availability (or not) of
  20. optional features such as int64 support.
  21. Each test group defines a single state object which gets applied as
  22. the test functions' `this` for all tests in that group. Test
  23. functions can use that to, e.g., set up a db in an early test and
  24. close it in a later test. Each test gets passed the sqlite3
  25. namespace object as its only argument.
  26. */
  27. /*
  28. This file is intended to be processed by c-pp to inject (or not)
  29. code specific to ES6 modules which is illegal in non-module code.
  30. Non-ES6 module build and ES6 module for the main-thread:
  31. ./c-pp -f tester1.c-pp.js -o tester1.js
  32. ES6 worker module build:
  33. ./c-pp -f tester1.c-pp.js -o tester1-esm.js -Dtarget=es6-module
  34. */
  35. //#if target=es6-module
  36. import {default as sqlite3InitModule} from './jswasm/sqlite3.mjs';
  37. globalThis.sqlite3InitModule = sqlite3InitModule;
  38. //#else
  39. 'use strict';
  40. //#endif
  41. (function(self){
  42. /**
  43. Set up our output channel differently depending
  44. on whether we are running in a worker thread or
  45. the main (UI) thread.
  46. */
  47. let logClass;
  48. /* Predicate for tests/groups. */
  49. const isUIThread = ()=>(globalThis.window===self && globalThis.document);
  50. /* Predicate for tests/groups. */
  51. const isWorker = ()=>!isUIThread();
  52. /* Predicate for tests/groups. */
  53. const testIsTodo = ()=>false;
  54. const haveWasmCTests = ()=>{
  55. return !!wasm.exports.sqlite3__wasm_test_intptr;
  56. };
  57. const hasOpfs = ()=>{
  58. return globalThis.FileSystemHandle
  59. && globalThis.FileSystemDirectoryHandle
  60. && globalThis.FileSystemFileHandle
  61. && globalThis.FileSystemFileHandle.prototype.createSyncAccessHandle
  62. && navigator?.storage?.getDirectory;
  63. };
  64. {
  65. const mapToString = (v)=>{
  66. switch(typeof v){
  67. case 'number': case 'string': case 'boolean':
  68. case 'undefined': case 'bigint':
  69. return ''+v;
  70. default: break;
  71. }
  72. if(null===v) return 'null';
  73. if(v instanceof Error){
  74. v = {
  75. message: v.message,
  76. stack: v.stack,
  77. errorClass: v.name
  78. };
  79. }
  80. return JSON.stringify(v,undefined,2);
  81. };
  82. const normalizeArgs = (args)=>args.map(mapToString);
  83. if( isUIThread() ){
  84. console.log("Running in the UI thread.");
  85. const logTarget = document.querySelector('#test-output');
  86. logClass = function(cssClass,...args){
  87. const ln = document.createElement('div');
  88. if(cssClass){
  89. for(const c of (Array.isArray(cssClass) ? cssClass : [cssClass])){
  90. ln.classList.add(c);
  91. }
  92. }
  93. ln.append(document.createTextNode(normalizeArgs(args).join(' ')));
  94. logTarget.append(ln);
  95. };
  96. const cbReverse = document.querySelector('#cb-log-reverse');
  97. //cbReverse.setAttribute('checked','checked');
  98. const cbReverseKey = 'tester1:cb-log-reverse';
  99. const cbReverseIt = ()=>{
  100. logTarget.classList[cbReverse.checked ? 'add' : 'remove']('reverse');
  101. //localStorage.setItem(cbReverseKey, cbReverse.checked ? 1 : 0);
  102. };
  103. cbReverse.addEventListener('change', cbReverseIt, true);
  104. /*if(localStorage.getItem(cbReverseKey)){
  105. cbReverse.checked = !!(+localStorage.getItem(cbReverseKey));
  106. }*/
  107. cbReverseIt();
  108. }else{ /* Worker thread */
  109. console.log("Running in a Worker thread.");
  110. logClass = function(cssClass,...args){
  111. postMessage({
  112. type:'log',
  113. payload:{cssClass, args: normalizeArgs(args)}
  114. });
  115. };
  116. }
  117. }
  118. const reportFinalTestStatus = function(pass){
  119. if(isUIThread()){
  120. let e = document.querySelector('#color-target');
  121. e.classList.add(pass ? 'tests-pass' : 'tests-fail');
  122. e = document.querySelector('title');
  123. e.innerText = (pass ? 'PASS' : 'FAIL') + ': ' + e.innerText;
  124. }else{
  125. postMessage({type:'test-result', payload:{pass}});
  126. }
  127. };
  128. const log = (...args)=>{
  129. //console.log(...args);
  130. logClass('',...args);
  131. }
  132. const warn = (...args)=>{
  133. console.warn(...args);
  134. logClass('warning',...args);
  135. }
  136. const error = (...args)=>{
  137. console.error(...args);
  138. logClass('error',...args);
  139. };
  140. const toss = (...args)=>{
  141. error(...args);
  142. throw new Error(args.join(' '));
  143. };
  144. const tossQuietly = (...args)=>{
  145. throw new Error(args.join(' '));
  146. };
  147. const roundMs = (ms)=>Math.round(ms*100)/100;
  148. /**
  149. Helpers for writing sqlite3-specific tests.
  150. */
  151. const TestUtil = {
  152. /** Running total of the number of tests run via
  153. this API. */
  154. counter: 0,
  155. /**
  156. If expr is a function, it is called and its result
  157. is returned, coerced to a bool, else expr, coerced to
  158. a bool, is returned.
  159. */
  160. toBool: function(expr){
  161. return (expr instanceof Function) ? !!expr() : !!expr;
  162. },
  163. /** Throws if expr is false. If expr is a function, it is called
  164. and its result is evaluated. If passed multiple arguments,
  165. those after the first are a message string which get applied
  166. as an exception message if the assertion fails. The message
  167. arguments are concatenated together with a space between each.
  168. */
  169. assert: function f(expr, ...msg){
  170. ++this.counter;
  171. if(!this.toBool(expr)){
  172. throw new Error(msg.length ? msg.join(' ') : "Assertion failed.");
  173. }
  174. return this;
  175. },
  176. /** Calls f() and squelches any exception it throws. If it
  177. does not throw, this function throws. */
  178. mustThrow: function(f, msg){
  179. ++this.counter;
  180. let err;
  181. try{ f(); } catch(e){err=e;}
  182. if(!err) throw new Error(msg || "Expected exception.");
  183. return this;
  184. },
  185. /**
  186. Works like mustThrow() but expects filter to be a regex,
  187. function, or string to match/filter the resulting exception
  188. against. If f() does not throw, this test fails and an Error is
  189. thrown. If filter is a regex, the test passes if
  190. filter.test(error.message) passes. If it's a function, the test
  191. passes if filter(error) returns truthy. If it's a string, the
  192. test passes if the filter matches the exception message
  193. precisely. In all other cases the test fails, throwing an
  194. Error.
  195. If it throws, msg is used as the error report unless it's falsy,
  196. in which case a default is used.
  197. */
  198. mustThrowMatching: function(f, filter, msg){
  199. ++this.counter;
  200. let err;
  201. try{ f(); } catch(e){err=e;}
  202. if(!err) throw new Error(msg || "Expected exception.");
  203. let pass = false;
  204. if(filter instanceof RegExp) pass = filter.test(err.message);
  205. else if(filter instanceof Function) pass = filter(err);
  206. else if('string' === typeof filter) pass = (err.message === filter);
  207. if(!pass){
  208. throw new Error(msg || ("Filter rejected this exception: "+err.message));
  209. }
  210. return this;
  211. },
  212. /** Throws if expr is truthy or expr is a function and expr()
  213. returns truthy. */
  214. throwIf: function(expr, msg){
  215. ++this.counter;
  216. if(this.toBool(expr)) throw new Error(msg || "throwIf() failed");
  217. return this;
  218. },
  219. /** Throws if expr is falsy or expr is a function and expr()
  220. returns falsy. */
  221. throwUnless: function(expr, msg){
  222. ++this.counter;
  223. if(!this.toBool(expr)) throw new Error(msg || "throwUnless() failed");
  224. return this;
  225. },
  226. eqApprox: (v1,v2,factor=0.05)=>(v1>=(v2-factor) && v1<=(v2+factor)),
  227. TestGroup: (function(){
  228. let groupCounter = 0;
  229. const TestGroup = function(name, predicate){
  230. this.number = ++groupCounter;
  231. this.name = name;
  232. this.predicate = predicate;
  233. this.tests = [];
  234. };
  235. TestGroup.prototype = {
  236. addTest: function(testObj){
  237. this.tests.push(testObj);
  238. return this;
  239. },
  240. run: async function(sqlite3){
  241. logClass('group-start',"Group #"+this.number+':',this.name);
  242. if(this.predicate){
  243. const p = this.predicate(sqlite3);
  244. if(!p || 'string'===typeof p){
  245. logClass(['warning','skipping-group'],
  246. "SKIPPING group:", p ? p : "predicate says to" );
  247. return;
  248. }
  249. }
  250. const assertCount = TestUtil.counter;
  251. const groupState = Object.create(null);
  252. const skipped = [];
  253. let runtime = 0, i = 0;
  254. for(const t of this.tests){
  255. ++i;
  256. const n = this.number+"."+i;
  257. logClass('one-test-line', n+":", t.name);
  258. if(t.predicate){
  259. const p = t.predicate(sqlite3);
  260. if(!p || 'string'===typeof p){
  261. logClass(['warning','skipping-test'],
  262. "SKIPPING:", p ? p : "predicate says to" );
  263. skipped.push( n+': '+t.name );
  264. continue;
  265. }
  266. }
  267. const tc = TestUtil.counter, now = performance.now();
  268. let rc = t.test.call(groupState, sqlite3);
  269. /*if(rc instanceof Promise){
  270. rc = rc.catch((e)=>{
  271. error("Test failure:",e);
  272. throw e;
  273. });
  274. }*/
  275. await rc;
  276. const then = performance.now();
  277. runtime += then - now;
  278. logClass(['faded','one-test-summary'],
  279. TestUtil.counter - tc, 'assertion(s) in',
  280. roundMs(then-now),'ms');
  281. }
  282. logClass(['green','group-end'],
  283. "#"+this.number+":",
  284. (TestUtil.counter - assertCount),
  285. "assertion(s) in",roundMs(runtime),"ms");
  286. if(0 && skipped.length){
  287. logClass('warning',"SKIPPED test(s) in group",this.number+":",skipped);
  288. }
  289. }
  290. };
  291. return TestGroup;
  292. })()/*TestGroup*/,
  293. testGroups: [],
  294. currentTestGroup: undefined,
  295. addGroup: function(name, predicate){
  296. this.testGroups.push( this.currentTestGroup =
  297. new this.TestGroup(name, predicate) );
  298. return this;
  299. },
  300. addTest: function(name, callback){
  301. let predicate;
  302. if(1===arguments.length){
  303. this.currentTestGroup.addTest(arguments[0]);
  304. }else{
  305. this.currentTestGroup.addTest({
  306. name, predicate, test: callback
  307. });
  308. }
  309. return this;
  310. },
  311. runTests: async function(sqlite3){
  312. return new Promise(async function(pok,pnok){
  313. try {
  314. let runtime = 0;
  315. for(let g of this.testGroups){
  316. const now = performance.now();
  317. await g.run(sqlite3);
  318. runtime += performance.now() - now;
  319. }
  320. logClass(['strong','green','full-test-summary'],
  321. "Done running tests.",TestUtil.counter,"assertions in",
  322. roundMs(runtime),'ms');
  323. pok();
  324. reportFinalTestStatus(true);
  325. }catch(e){
  326. error(e);
  327. pnok(e);
  328. reportFinalTestStatus(false);
  329. }
  330. }.bind(this));
  331. }
  332. }/*TestUtil*/;
  333. const T = TestUtil;
  334. T.g = T.addGroup;
  335. T.t = T.addTest;
  336. let capi, wasm/*assigned after module init*/;
  337. const sahPoolConfig = {
  338. name: 'opfs-sahpool-tester1',
  339. clearOnInit: true,
  340. initialCapacity: 6
  341. };
  342. ////////////////////////////////////////////////////////////////////////
  343. // End of infrastructure setup. Now define the tests...
  344. ////////////////////////////////////////////////////////////////////////
  345. ////////////////////////////////////////////////////////////////////
  346. T.g('Basic sanity checks')
  347. .t({
  348. name:'sqlite3_config()',
  349. test:function(sqlite3){
  350. for(const k of [
  351. 'SQLITE_CONFIG_GETMALLOC', 'SQLITE_CONFIG_URI'
  352. ]){
  353. T.assert(capi[k] > 0);
  354. }
  355. T.assert(capi.SQLITE_MISUSE===capi.sqlite3_config(
  356. capi.SQLITE_CONFIG_URI, 1
  357. ), "MISUSE because the library has already been initialized.");
  358. T.assert(capi.SQLITE_MISUSE === capi.sqlite3_config(
  359. // not enough args
  360. capi.SQLITE_CONFIG_GETMALLOC
  361. ));
  362. T.assert(capi.SQLITE_NOTFOUND === capi.sqlite3_config(
  363. // unhandled-in-JS config option
  364. capi.SQLITE_CONFIG_GETMALLOC, 1
  365. ));
  366. if(0){
  367. log("We cannot _fully_ test sqlite3_config() after the library",
  368. "has been initialized (which it necessarily has been to",
  369. "set up various bindings) and we cannot shut it down ",
  370. "without losing the VFS registrations.");
  371. T.assert(0 === capi.sqlite3_config(
  372. capi.SQLITE_CONFIG_URI, 1
  373. ));
  374. }
  375. }
  376. })/*sqlite3_config()*/
  377. ////////////////////////////////////////////////////////////////////
  378. .t({
  379. name: "JS wasm-side allocator",
  380. test: function(sqlite3){
  381. if(sqlite3.config.useStdAlloc){
  382. warn("Using system allocator. This violates the docs and",
  383. "may cause grief with certain APIs",
  384. "(e.g. sqlite3_deserialize()).");
  385. T.assert(wasm.alloc.impl === wasm.exports.malloc)
  386. .assert(wasm.dealloc === wasm.exports.free)
  387. .assert(wasm.realloc.impl === wasm.exports.realloc);
  388. }else{
  389. T.assert(wasm.alloc.impl === wasm.exports.sqlite3_malloc)
  390. .assert(wasm.dealloc === wasm.exports.sqlite3_free)
  391. .assert(wasm.realloc.impl === wasm.exports.sqlite3_realloc);
  392. }
  393. }
  394. })
  395. .t('Namespace object checks', function(sqlite3){
  396. const wasmCtypes = wasm.ctype;
  397. T.assert(wasmCtypes.structs[0].name==='sqlite3_vfs').
  398. assert(wasmCtypes.structs[0].members.szOsFile.sizeof>=4).
  399. assert(wasmCtypes.structs[1/*sqlite3_io_methods*/
  400. ].members.xFileSize.offset>0);
  401. [ /* Spot-check a handful of constants to make sure they got installed... */
  402. 'SQLITE_SCHEMA','SQLITE_NULL','SQLITE_UTF8',
  403. 'SQLITE_STATIC', 'SQLITE_DIRECTONLY',
  404. 'SQLITE_OPEN_CREATE', 'SQLITE_OPEN_DELETEONCLOSE'
  405. ].forEach((k)=>T.assert('number' === typeof capi[k]));
  406. [/* Spot-check a few of the WASM API methods. */
  407. 'alloc', 'dealloc', 'installFunction'
  408. ].forEach((k)=>T.assert(wasm[k] instanceof Function));
  409. T.assert(capi.sqlite3_errstr(capi.SQLITE_IOERR_ACCESS).indexOf("I/O")>=0).
  410. assert(capi.sqlite3_errstr(capi.SQLITE_CORRUPT).indexOf('malformed')>0).
  411. assert(capi.sqlite3_errstr(capi.SQLITE_OK) === 'not an error');
  412. try {
  413. throw new sqlite3.WasmAllocError;
  414. }catch(e){
  415. T.assert(e instanceof Error)
  416. .assert(e instanceof sqlite3.WasmAllocError)
  417. .assert("Allocation failed." === e.message);
  418. }
  419. try {
  420. throw new sqlite3.WasmAllocError("test",{
  421. cause: 3
  422. });
  423. }catch(e){
  424. T.assert(3 === e.cause)
  425. .assert("test" === e.message);
  426. }
  427. try {throw new sqlite3.WasmAllocError("test","ing",".")}
  428. catch(e){T.assert("test ing ." === e.message)}
  429. try{ throw new sqlite3.SQLite3Error(capi.SQLITE_SCHEMA) }
  430. catch(e){
  431. T.assert('SQLITE_SCHEMA' === e.message)
  432. .assert(capi.SQLITE_SCHEMA === e.resultCode);
  433. }
  434. try{ sqlite3.SQLite3Error.toss(capi.SQLITE_CORRUPT,{cause: true}) }
  435. catch(e){
  436. T.assert('SQLITE_CORRUPT' === e.message)
  437. .assert(capi.SQLITE_CORRUPT === e.resultCode)
  438. .assert(true===e.cause);
  439. }
  440. try{ sqlite3.SQLite3Error.toss("resultCode check") }
  441. catch(e){
  442. T.assert(capi.SQLITE_ERROR === e.resultCode)
  443. .assert('resultCode check' === e.message);
  444. }
  445. })
  446. ////////////////////////////////////////////////////////////////////
  447. .t('strglob/strlike', function(sqlite3){
  448. T.assert(0===capi.sqlite3_strglob("*.txt", "foo.txt")).
  449. assert(0!==capi.sqlite3_strglob("*.txt", "foo.xtx")).
  450. assert(0===capi.sqlite3_strlike("%.txt", "foo.txt", 0)).
  451. assert(0!==capi.sqlite3_strlike("%.txt", "foo.xtx", 0));
  452. })
  453. ////////////////////////////////////////////////////////////////////
  454. ;/*end of basic sanity checks*/
  455. ////////////////////////////////////////////////////////////////////
  456. T.g('C/WASM Utilities')
  457. .t('sqlite3.wasm namespace', function(sqlite3){
  458. // TODO: break this into smaller individual test functions.
  459. const w = wasm;
  460. const chr = (x)=>x.charCodeAt(0);
  461. //log("heap getters...");
  462. {
  463. const li = [8, 16, 32];
  464. if(w.bigIntEnabled) li.push(64);
  465. for(const n of li){
  466. const bpe = n/8;
  467. const s = w.heapForSize(n,false);
  468. T.assert(bpe===s.BYTES_PER_ELEMENT).
  469. assert(w.heapForSize(s.constructor) === s);
  470. const u = w.heapForSize(n,true);
  471. T.assert(bpe===u.BYTES_PER_ELEMENT).
  472. assert(s!==u).
  473. assert(w.heapForSize(u.constructor) === u);
  474. }
  475. }
  476. // alloc(), realloc(), allocFromTypedArray()
  477. {
  478. let m = w.alloc(14);
  479. let m2 = w.realloc(m, 16);
  480. T.assert(m === m2/* because of alignment */);
  481. T.assert(0 === w.realloc(m, 0));
  482. m = m2 = 0;
  483. // Check allocation limits and allocator's responses...
  484. T.assert('number' === typeof sqlite3.capi.SQLITE_MAX_ALLOCATION_SIZE);
  485. if(!sqlite3.config.useStdAlloc){
  486. const tooMuch = sqlite3.capi.SQLITE_MAX_ALLOCATION_SIZE + 1,
  487. isAllocErr = (e)=>e instanceof sqlite3.WasmAllocError;
  488. T.mustThrowMatching(()=>w.alloc(tooMuch), isAllocErr)
  489. .assert(0 === w.alloc.impl(tooMuch))
  490. .mustThrowMatching(()=>w.realloc(0, tooMuch), isAllocErr)
  491. .assert(0 === w.realloc.impl(0, tooMuch));
  492. }
  493. // Check allocFromTypedArray()...
  494. const byteList = [11,22,33]
  495. const u = new Uint8Array(byteList);
  496. m = w.allocFromTypedArray(u);
  497. for(let i = 0; i < u.length; ++i){
  498. T.assert(u[i] === byteList[i])
  499. .assert(u[i] === w.peek8(m + i));
  500. }
  501. w.dealloc(m);
  502. m = w.allocFromTypedArray(u.buffer);
  503. for(let i = 0; i < u.length; ++i){
  504. T.assert(u[i] === byteList[i])
  505. .assert(u[i] === w.peek8(m + i));
  506. }
  507. w.dealloc(m);
  508. T.mustThrowMatching(
  509. ()=>w.allocFromTypedArray(1),
  510. 'Value is not of a supported TypedArray type.'
  511. );
  512. }
  513. { // Test peekXYZ()/pokeXYZ()...
  514. const m = w.alloc(8);
  515. T.assert( 17 === w.poke8(m,17).peek8(m) )
  516. .assert( 31987 === w.poke16(m,31987).peek16(m) )
  517. .assert( 345678 === w.poke32(m,345678).peek32(m) )
  518. .assert(
  519. T.eqApprox( 345678.9, w.poke32f(m,345678.9).peek32f(m) )
  520. ).assert(
  521. T.eqApprox( 4567890123.4, w.poke64f(m, 4567890123.4).peek64f(m) )
  522. );
  523. if(w.bigIntEnabled){
  524. T.assert(
  525. BigInt(Number.MAX_SAFE_INTEGER) ===
  526. w.poke64(m, Number.MAX_SAFE_INTEGER).peek64(m)
  527. );
  528. }
  529. w.dealloc(m);
  530. }
  531. // isPtr32()
  532. {
  533. const ip = w.isPtr32;
  534. T.assert(ip(0))
  535. .assert(!ip(-1))
  536. .assert(!ip(1.1))
  537. .assert(!ip(0xffffffff))
  538. .assert(ip(0x7fffffff))
  539. .assert(!ip())
  540. .assert(!ip(null)/*might change: under consideration*/)
  541. ;
  542. }
  543. //log("jstrlen()...");
  544. {
  545. T.assert(3 === w.jstrlen("abc")).assert(4 === w.jstrlen("äbc"));
  546. }
  547. //log("jstrcpy()...");
  548. {
  549. const fillChar = 10;
  550. let ua = new Uint8Array(8), rc,
  551. refill = ()=>ua.fill(fillChar);
  552. refill();
  553. rc = w.jstrcpy("hello", ua);
  554. T.assert(6===rc).assert(0===ua[5]).assert(chr('o')===ua[4]);
  555. refill();
  556. ua[5] = chr('!');
  557. rc = w.jstrcpy("HELLO", ua, 0, -1, false);
  558. T.assert(5===rc).assert(chr('!')===ua[5]).assert(chr('O')===ua[4]);
  559. refill();
  560. rc = w.jstrcpy("the end", ua, 4);
  561. //log("rc,ua",rc,ua);
  562. T.assert(4===rc).assert(0===ua[7]).
  563. assert(chr('e')===ua[6]).assert(chr('t')===ua[4]);
  564. refill();
  565. rc = w.jstrcpy("the end", ua, 4, -1, false);
  566. T.assert(4===rc).assert(chr(' ')===ua[7]).
  567. assert(chr('e')===ua[6]).assert(chr('t')===ua[4]);
  568. refill();
  569. rc = w.jstrcpy("", ua, 0, 1, true);
  570. //log("rc,ua",rc,ua);
  571. T.assert(1===rc).assert(0===ua[0]);
  572. refill();
  573. rc = w.jstrcpy("x", ua, 0, 1, true);
  574. //log("rc,ua",rc,ua);
  575. T.assert(1===rc).assert(0===ua[0]);
  576. refill();
  577. rc = w.jstrcpy('äbä', ua, 0, 1, true);
  578. T.assert(1===rc, 'Must not write partial multi-byte char.')
  579. .assert(0===ua[0]);
  580. refill();
  581. rc = w.jstrcpy('äbä', ua, 0, 2, true);
  582. T.assert(1===rc, 'Must not write partial multi-byte char.')
  583. .assert(0===ua[0]);
  584. refill();
  585. rc = w.jstrcpy('äbä', ua, 0, 2, false);
  586. T.assert(2===rc).assert(fillChar!==ua[1]).assert(fillChar===ua[2]);
  587. }/*jstrcpy()*/
  588. //log("cstrncpy()...");
  589. {
  590. const scope = w.scopedAllocPush();
  591. try {
  592. let cStr = w.scopedAllocCString("hello");
  593. const n = w.cstrlen(cStr);
  594. let cpy = w.scopedAlloc(n+10);
  595. let rc = w.cstrncpy(cpy, cStr, n+10);
  596. T.assert(n+1 === rc).
  597. assert("hello" === w.cstrToJs(cpy)).
  598. assert(chr('o') === w.peek8(cpy+n-1)).
  599. assert(0 === w.peek8(cpy+n));
  600. let cStr2 = w.scopedAllocCString("HI!!!");
  601. rc = w.cstrncpy(cpy, cStr2, 3);
  602. T.assert(3===rc).
  603. assert("HI!lo" === w.cstrToJs(cpy)).
  604. assert(chr('!') === w.peek8(cpy+2)).
  605. assert(chr('l') === w.peek8(cpy+3));
  606. }finally{
  607. w.scopedAllocPop(scope);
  608. }
  609. }
  610. //log("jstrToUintArray()...");
  611. {
  612. let a = w.jstrToUintArray("hello", false);
  613. T.assert(5===a.byteLength).assert(chr('o')===a[4]);
  614. a = w.jstrToUintArray("hello", true);
  615. T.assert(6===a.byteLength).assert(chr('o')===a[4]).assert(0===a[5]);
  616. a = w.jstrToUintArray("äbä", false);
  617. T.assert(5===a.byteLength).assert(chr('b')===a[2]);
  618. a = w.jstrToUintArray("äbä", true);
  619. T.assert(6===a.byteLength).assert(chr('b')===a[2]).assert(0===a[5]);
  620. }
  621. //log("allocCString()...");
  622. {
  623. const jstr = "hällo, world!";
  624. const [cstr, n] = w.allocCString(jstr, true);
  625. T.assert(14 === n)
  626. .assert(0===w.peek8(cstr+n))
  627. .assert(chr('!')===w.peek8(cstr+n-1));
  628. w.dealloc(cstr);
  629. }
  630. //log("scopedAlloc() and friends...");
  631. {
  632. const alloc = w.alloc, dealloc = w.dealloc;
  633. w.alloc = w.dealloc = null;
  634. T.assert(!w.scopedAlloc.level)
  635. .mustThrowMatching(()=>w.scopedAlloc(1), /^No scopedAllocPush/)
  636. .mustThrowMatching(()=>w.scopedAllocPush(), /missing alloc/);
  637. w.alloc = alloc;
  638. T.mustThrowMatching(()=>w.scopedAllocPush(), /missing alloc/);
  639. w.dealloc = dealloc;
  640. T.mustThrowMatching(()=>w.scopedAllocPop(), /^Invalid state/)
  641. .mustThrowMatching(()=>w.scopedAlloc(1), /^No scopedAllocPush/)
  642. .mustThrowMatching(()=>w.scopedAlloc.level=0, /read-only/);
  643. const asc = w.scopedAllocPush();
  644. let asc2;
  645. try {
  646. const p1 = w.scopedAlloc(16),
  647. p2 = w.scopedAlloc(16);
  648. T.assert(1===w.scopedAlloc.level)
  649. .assert(Number.isFinite(p1))
  650. .assert(Number.isFinite(p2))
  651. .assert(asc[0] === p1)
  652. .assert(asc[1]===p2);
  653. asc2 = w.scopedAllocPush();
  654. const p3 = w.scopedAlloc(16);
  655. T.assert(2===w.scopedAlloc.level)
  656. .assert(Number.isFinite(p3))
  657. .assert(2===asc.length)
  658. .assert(p3===asc2[0]);
  659. const [z1, z2, z3] = w.scopedAllocPtr(3);
  660. T.assert('number'===typeof z1).assert(z2>z1).assert(z3>z2)
  661. .assert(0===w.peek32(z1), 'allocPtr() must zero the targets')
  662. .assert(0===w.peek32(z3));
  663. }finally{
  664. // Pop them in "incorrect" order to make sure they behave:
  665. w.scopedAllocPop(asc);
  666. T.assert(0===asc.length);
  667. T.mustThrowMatching(()=>w.scopedAllocPop(asc),
  668. /^Invalid state object/);
  669. if(asc2){
  670. T.assert(2===asc2.length,'Should be p3 and z1');
  671. w.scopedAllocPop(asc2);
  672. T.assert(0===asc2.length);
  673. T.mustThrowMatching(()=>w.scopedAllocPop(asc2),
  674. /^Invalid state object/);
  675. }
  676. }
  677. T.assert(0===w.scopedAlloc.level);
  678. w.scopedAllocCall(function(){
  679. T.assert(1===w.scopedAlloc.level);
  680. const [cstr, n] = w.scopedAllocCString("hello, world", true);
  681. T.assert(12 === n)
  682. .assert(0===w.peek8(cstr+n))
  683. .assert(chr('d')===w.peek8(cstr+n-1));
  684. });
  685. }/*scopedAlloc()*/
  686. //log("xCall()...");
  687. {
  688. const pJson = w.xCall('sqlite3__wasm_enum_json');
  689. T.assert(Number.isFinite(pJson)).assert(w.cstrlen(pJson)>300);
  690. }
  691. //log("xWrap()...");
  692. {
  693. T.mustThrowMatching(()=>w.xWrap('sqlite3_libversion',null,'i32'),
  694. /requires 0 arg/).
  695. assert(w.xWrap.resultAdapter('i32') instanceof Function).
  696. assert(w.xWrap.argAdapter('i32') instanceof Function);
  697. let fw = w.xWrap('sqlite3_libversion','utf8');
  698. T.mustThrowMatching(()=>fw(1), /requires 0 arg/);
  699. let rc = fw();
  700. T.assert('string'===typeof rc).assert(rc.length>5);
  701. rc = w.xCallWrapped('sqlite3__wasm_enum_json','*');
  702. T.assert(rc>0 && Number.isFinite(rc));
  703. rc = w.xCallWrapped('sqlite3__wasm_enum_json','utf8');
  704. T.assert('string'===typeof rc).assert(rc.length>300);
  705. { // 'string:static' argAdapter() sanity checks...
  706. let argAd = w.xWrap.argAdapter('string:static');
  707. let p0 = argAd('foo'), p1 = argAd('bar');
  708. T.assert(w.isPtr(p0) && w.isPtr(p1))
  709. .assert(p0 !== p1)
  710. .assert(p0 === argAd('foo'))
  711. .assert(p1 === argAd('bar'));
  712. }
  713. // 'string:flexible' argAdapter() sanity checks...
  714. w.scopedAllocCall(()=>{
  715. const argAd = w.xWrap.argAdapter('string:flexible');
  716. const cj = (v)=>w.cstrToJs(argAd(v));
  717. T.assert('Hi' === cj('Hi'))
  718. .assert('hi' === cj(['h','i']))
  719. .assert('HI' === cj(new Uint8Array([72, 73])));
  720. });
  721. // jsFuncToWasm()
  722. {
  723. const fsum3 = (x,y,z)=>x+y+z;
  724. fw = w.jsFuncToWasm('i(iii)', fsum3);
  725. T.assert(fw instanceof Function)
  726. .assert( fsum3 !== fw )
  727. .assert( 3 === fw.length )
  728. .assert( 6 === fw(1,2,3) );
  729. T.mustThrowMatching( ()=>w.jsFuncToWasm('x()', function(){}),
  730. 'Invalid signature letter: x');
  731. }
  732. // xWrap(Function,...)
  733. {
  734. let fp;
  735. try {
  736. const fmy = function fmy(i,s,d){
  737. if(fmy.debug) log("fmy(",...arguments,")");
  738. T.assert( 3 === i )
  739. .assert( w.isPtr(s) )
  740. .assert( w.cstrToJs(s) === 'a string' )
  741. .assert( T.eqApprox(1.2, d) );
  742. return w.allocCString("hi");
  743. };
  744. fmy.debug = false;
  745. const xwArgs = ['string:dealloc', ['i32', 'string', 'f64']];
  746. fw = w.xWrap(fmy, ...xwArgs);
  747. const fmyArgs = [3, 'a string', 1.2];
  748. let rc = fw(...fmyArgs);
  749. T.assert( 'hi' === rc );
  750. if(0){
  751. /* Retain this as a "reminder to self"...
  752. This extra level of indirection does not work: the
  753. string argument is ending up as a null in fmy() but
  754. the numeric arguments are making their ways through
  755. What's happening is: installFunction() is creating a
  756. WASM-compatible function instance. When we pass a JS string
  757. into there it's getting coerced into `null` before being passed
  758. on to the lower-level wrapper.
  759. */
  760. fmy.debug = true;
  761. fp = wasm.installFunction('i(isd)', fw);
  762. fw = w.functionEntry(fp);
  763. rc = fw(...fmyArgs);
  764. log("rc =",rc);
  765. T.assert( 'hi' === rc );
  766. // Similarly, this does not work:
  767. //let fpw = w.xWrap(fp, null, [null,null,null]);
  768. //rc = fpw(...fmyArgs);
  769. //log("rc =",rc);
  770. //T.assert( 'hi' === rc );
  771. }
  772. }finally{
  773. wasm.uninstallFunction(fp);
  774. }
  775. }
  776. if(haveWasmCTests()){
  777. if(!sqlite3.config.useStdAlloc){
  778. fw = w.xWrap('sqlite3__wasm_test_str_hello', 'utf8:dealloc',['i32']);
  779. rc = fw(0);
  780. T.assert('hello'===rc);
  781. rc = fw(1);
  782. T.assert(null===rc);
  783. }
  784. if(w.bigIntEnabled){
  785. w.xWrap.resultAdapter('thrice', (v)=>3n*BigInt(v));
  786. w.xWrap.argAdapter('twice', (v)=>2n*BigInt(v));
  787. fw = w.xWrap('sqlite3__wasm_test_int64_times2','thrice','twice');
  788. rc = fw(1);
  789. T.assert(12n===rc);
  790. w.scopedAllocCall(function(){
  791. const pI1 = w.scopedAlloc(8), pI2 = pI1+4;
  792. w.pokePtr([pI1, pI2], 0);
  793. const f = w.xWrap('sqlite3__wasm_test_int64_minmax',undefined,['i64*','i64*']);
  794. const [r1, r2] = w.peek64([pI1, pI2]);
  795. T.assert(!Number.isSafeInteger(r1)).assert(!Number.isSafeInteger(r2));
  796. });
  797. }
  798. }
  799. }/*xWrap()*/
  800. }/*WhWasmUtil*/)
  801. ////////////////////////////////////////////////////////////////////
  802. .t({
  803. name: 'sqlite3.StructBinder (jaccwabyt🐇)',
  804. predicate: (sqlite3)=>!!sqlite3.wasm.exports.sqlite3__wasm_test_struct
  805. || "Built without SQLITE_WASM_ENABLE_C_TESTS",
  806. test: function(sqlite3){
  807. const S = sqlite3, W = S.wasm;
  808. const MyStructDef = {
  809. sizeof: 16,
  810. members: {
  811. p4: {offset: 0, sizeof: 4, signature: "i"},
  812. pP: {offset: 4, sizeof: 4, signature: "P"},
  813. ro: {offset: 8, sizeof: 4, signature: "i", readOnly: true},
  814. cstr: {offset: 12, sizeof: 4, signature: "s"}
  815. }
  816. };
  817. if(W.bigIntEnabled){
  818. const m = MyStructDef;
  819. m.members.p8 = {offset: m.sizeof, sizeof: 8, signature: "j"};
  820. m.sizeof += m.members.p8.sizeof;
  821. }
  822. const StructType = S.StructBinder.StructType;
  823. const K = S.StructBinder('my_struct',MyStructDef);
  824. T.mustThrowMatching(()=>K(), /via 'new'/).
  825. mustThrowMatching(()=>new K('hi'), /^Invalid pointer/);
  826. const k1 = new K(), k2 = new K();
  827. try {
  828. T.assert(k1.constructor === K).
  829. assert(K.isA(k1)).
  830. assert(k1 instanceof K).
  831. assert(K.prototype.lookupMember('p4').key === '$p4').
  832. assert(K.prototype.lookupMember('$p4').name === 'p4').
  833. mustThrowMatching(()=>K.prototype.lookupMember('nope'), /not a mapped/).
  834. assert(undefined === K.prototype.lookupMember('nope',false)).
  835. assert(k1 instanceof StructType).
  836. assert(StructType.isA(k1)).
  837. mustThrowMatching(()=>k1.$ro = 1, /read-only/);
  838. Object.keys(MyStructDef.members).forEach(function(key){
  839. key = K.memberKey(key);
  840. T.assert(0 == k1[key],
  841. "Expecting allocation to zero the memory "+
  842. "for "+key+" but got: "+k1[key]+
  843. " from "+k1.memoryDump());
  844. });
  845. T.assert('number' === typeof k1.pointer).
  846. mustThrowMatching(()=>k1.pointer = 1, /pointer/);
  847. k1.$p4 = 1; k1.$pP = 2;
  848. T.assert(1 === k1.$p4).assert(2 === k1.$pP);
  849. if(MyStructDef.members.$p8){
  850. k1.$p8 = 1/*must not throw despite not being a BigInt*/;
  851. k1.$p8 = BigInt(Number.MAX_SAFE_INTEGER * 2);
  852. T.assert(BigInt(2 * Number.MAX_SAFE_INTEGER) === k1.$p8);
  853. }
  854. T.assert(!k1.ondispose);
  855. k1.setMemberCString('cstr', "A C-string.");
  856. T.assert(Array.isArray(k1.ondispose)).
  857. assert(k1.ondispose[0] === k1.$cstr).
  858. assert('number' === typeof k1.$cstr).
  859. assert('A C-string.' === k1.memberToJsString('cstr'));
  860. k1.$pP = k2;
  861. T.assert(k1.$pP === k2.pointer);
  862. k1.$pP = null/*null is special-cased to 0.*/;
  863. T.assert(0===k1.$pP);
  864. let ptr = k1.pointer;
  865. k1.dispose();
  866. T.assert(undefined === k1.pointer).
  867. mustThrowMatching(()=>{k1.$pP=1}, /disposed instance/);
  868. }finally{
  869. k1.dispose();
  870. k2.dispose();
  871. }
  872. if(!W.bigIntEnabled){
  873. log("Skipping WasmTestStruct tests: BigInt not enabled.");
  874. return;
  875. }
  876. const WTStructDesc =
  877. W.ctype.structs.filter((e)=>'WasmTestStruct'===e.name)[0];
  878. const autoResolvePtr = true /* EXPERIMENTAL */;
  879. if(autoResolvePtr){
  880. WTStructDesc.members.ppV.signature = 'P';
  881. }
  882. const WTStruct = S.StructBinder(WTStructDesc);
  883. //log(WTStruct.structName, WTStruct.structInfo);
  884. const wts = new WTStruct();
  885. //log("WTStruct.prototype keys:",Object.keys(WTStruct.prototype));
  886. try{
  887. T.assert(wts.constructor === WTStruct).
  888. assert(WTStruct.memberKeys().indexOf('$ppV')>=0).
  889. assert(wts.memberKeys().indexOf('$v8')>=0).
  890. assert(!K.isA(wts)).
  891. assert(WTStruct.isA(wts)).
  892. assert(wts instanceof WTStruct).
  893. assert(wts instanceof StructType).
  894. assert(StructType.isA(wts)).
  895. assert(wts.pointer>0).assert(0===wts.$v4).assert(0n===wts.$v8).
  896. assert(0===wts.$ppV).assert(0===wts.$xFunc);
  897. const testFunc =
  898. W.xGet('sqlite3__wasm_test_struct'/*name gets mangled in -O3 builds!*/);
  899. let counter = 0;
  900. //log("wts.pointer =",wts.pointer);
  901. const wtsFunc = function(arg){
  902. /*log("This from a JS function called from C, "+
  903. "which itself was called from JS. arg =",arg);*/
  904. ++counter;
  905. if(3===counter){
  906. tossQuietly("Testing exception propagation.");
  907. }
  908. }
  909. wts.$v4 = 10; wts.$v8 = 20;
  910. wts.$xFunc = W.installFunction(wtsFunc, wts.memberSignature('xFunc'))
  911. T.assert(0===counter).assert(10 === wts.$v4).assert(20n === wts.$v8)
  912. .assert(0 === wts.$ppV).assert('number' === typeof wts.$xFunc)
  913. .assert(0 === wts.$cstr)
  914. .assert(wts.memberIsString('$cstr'))
  915. .assert(!wts.memberIsString('$v4'))
  916. .assert(null === wts.memberToJsString('$cstr'))
  917. .assert(W.functionEntry(wts.$xFunc) instanceof Function);
  918. /* It might seem silly to assert that the values match
  919. what we just set, but recall that all of those property
  920. reads and writes are, via property interceptors,
  921. actually marshaling their data to/from a raw memory
  922. buffer, so merely reading them back is actually part of
  923. testing the struct-wrapping API. */
  924. testFunc(wts.pointer);
  925. //log("wts.pointer, wts.$ppV",wts.pointer, wts.$ppV);
  926. T.assert(1===counter).assert(20 === wts.$v4).assert(40n === wts.$v8)
  927. .assert(wts.$ppV === wts.pointer)
  928. .assert('string' === typeof wts.memberToJsString('cstr'))
  929. .assert(wts.memberToJsString('cstr') === wts.memberToJsString('$cstr'))
  930. .mustThrowMatching(()=>wts.memberToJsString('xFunc'),
  931. /Invalid member type signature for C-string/)
  932. ;
  933. testFunc(wts.pointer);
  934. T.assert(2===counter).assert(40 === wts.$v4).assert(80n === wts.$v8)
  935. .assert(wts.$ppV === wts.pointer);
  936. /** The 3rd call to wtsFunc throw from JS, which is called
  937. from C, which is called from JS. Let's ensure that
  938. that exception propagates back here... */
  939. T.mustThrowMatching(()=>testFunc(wts.pointer),/^Testing/);
  940. W.uninstallFunction(wts.$xFunc);
  941. wts.$xFunc = 0;
  942. wts.$ppV = 0;
  943. T.assert(!wts.$ppV);
  944. //WTStruct.debugFlags(0x03);
  945. wts.$ppV = wts;
  946. T.assert(wts.pointer === wts.$ppV)
  947. wts.setMemberCString('cstr', "A C-string.");
  948. T.assert(Array.isArray(wts.ondispose)).
  949. assert(wts.ondispose[0] === wts.$cstr).
  950. assert('A C-string.' === wts.memberToJsString('cstr'));
  951. const ptr = wts.pointer;
  952. wts.dispose();
  953. T.assert(ptr).assert(undefined === wts.pointer);
  954. }finally{
  955. wts.dispose();
  956. }
  957. if(1){ // ondispose of other struct instances
  958. const s1 = new WTStruct, s2 = new WTStruct, s3 = new WTStruct;
  959. T.assert(s1.lookupMember instanceof Function)
  960. .assert(s1.addOnDispose instanceof Function);
  961. s1.addOnDispose(s2,"testing variadic args");
  962. T.assert(2===s1.ondispose.length);
  963. s2.addOnDispose(s3);
  964. s1.dispose();
  965. T.assert(!s2.pointer,"Expecting s2 to be ondispose'd by s1.");
  966. T.assert(!s3.pointer,"Expecting s3 to be ondispose'd by s2.");
  967. }
  968. }
  969. }/*StructBinder*/)
  970. ////////////////////////////////////////////////////////////////////
  971. .t('sqlite3.wasm.pstack', function(sqlite3){
  972. const P = wasm.pstack;
  973. const isAllocErr = (e)=>e instanceof sqlite3.WasmAllocError;
  974. const stack = P.pointer;
  975. T.assert(0===stack % 8 /* must be 8-byte aligned */);
  976. try{
  977. const remaining = P.remaining;
  978. T.assert(P.quota >= 4096)
  979. .assert(remaining === P.quota)
  980. .mustThrowMatching(()=>P.alloc(0), isAllocErr)
  981. .mustThrowMatching(()=>P.alloc(-1), isAllocErr)
  982. .mustThrowMatching(
  983. ()=>P.alloc('i33'),
  984. (e)=>e instanceof sqlite3.WasmAllocError
  985. );
  986. ;
  987. let p1 = P.alloc(12);
  988. T.assert(p1 === stack - 16/*8-byte aligned*/)
  989. .assert(P.pointer === p1);
  990. let p2 = P.alloc(7);
  991. T.assert(p2 === p1-8/*8-byte aligned, stack grows downwards*/)
  992. .mustThrowMatching(()=>P.alloc(remaining), isAllocErr)
  993. .assert(24 === stack - p2)
  994. .assert(P.pointer === p2);
  995. let n = remaining - (stack - p2);
  996. let p3 = P.alloc(n);
  997. T.assert(p3 === stack-remaining)
  998. .mustThrowMatching(()=>P.alloc(1), isAllocErr);
  999. }finally{
  1000. P.restore(stack);
  1001. }
  1002. T.assert(P.pointer === stack);
  1003. try {
  1004. const [p1, p2, p3] = P.allocChunks(3,'i32');
  1005. T.assert(P.pointer === stack-16/*always rounded to multiple of 8*/)
  1006. .assert(p2 === p1 + 4)
  1007. .assert(p3 === p2 + 4);
  1008. T.mustThrowMatching(()=>P.allocChunks(1024, 1024 * 16),
  1009. (e)=>e instanceof sqlite3.WasmAllocError)
  1010. }finally{
  1011. P.restore(stack);
  1012. }
  1013. T.assert(P.pointer === stack);
  1014. try {
  1015. let [p1, p2, p3] = P.allocPtr(3,false);
  1016. let sPos = stack-16/*always rounded to multiple of 8*/;
  1017. T.assert(P.pointer === sPos)
  1018. .assert(p2 === p1 + 4)
  1019. .assert(p3 === p2 + 4);
  1020. [p1, p2, p3] = P.allocPtr(3);
  1021. T.assert(P.pointer === sPos-24/*3 x 8 bytes*/)
  1022. .assert(p2 === p1 + 8)
  1023. .assert(p3 === p2 + 8);
  1024. p1 = P.allocPtr();
  1025. T.assert('number'===typeof p1);
  1026. }finally{
  1027. P.restore(stack);
  1028. }
  1029. }/*pstack tests*/)
  1030. ////////////////////////////////////////////////////////////////////
  1031. ;/*end of C/WASM utils checks*/
  1032. T.g('sqlite3_randomness()')
  1033. .t('To memory buffer', function(sqlite3){
  1034. const stack = wasm.pstack.pointer;
  1035. try{
  1036. const n = 520;
  1037. const p = wasm.pstack.alloc(n);
  1038. T.assert(0===wasm.peek8(p))
  1039. .assert(0===wasm.peek8(p+n-1));
  1040. T.assert(undefined === capi.sqlite3_randomness(n - 10, p));
  1041. let j, check = 0;
  1042. const heap = wasm.heap8u();
  1043. for(j = 0; j < 10 && 0===check; ++j){
  1044. check += heap[p + j];
  1045. }
  1046. T.assert(check > 0);
  1047. check = 0;
  1048. // Ensure that the trailing bytes were not modified...
  1049. for(j = n - 10; j < n && 0===check; ++j){
  1050. check += heap[p + j];
  1051. }
  1052. T.assert(0===check);
  1053. }finally{
  1054. wasm.pstack.restore(stack);
  1055. }
  1056. })
  1057. .t('To byte array', function(sqlite3){
  1058. const ta = new Uint8Array(117);
  1059. let i, n = 0;
  1060. for(i=0; i<ta.byteLength && 0===n; ++i){
  1061. n += ta[i];
  1062. }
  1063. T.assert(0===n)
  1064. .assert(ta === capi.sqlite3_randomness(ta));
  1065. for(i=ta.byteLength-10; i<ta.byteLength && 0===n; ++i){
  1066. n += ta[i];
  1067. }
  1068. T.assert(n>0);
  1069. const t0 = new Uint8Array(0);
  1070. T.assert(t0 === capi.sqlite3_randomness(t0),
  1071. "0-length array is a special case");
  1072. })
  1073. ;/*end sqlite3_randomness() checks*/
  1074. ////////////////////////////////////////////////////////////////////////
  1075. T.g('sqlite3.oo1')
  1076. .t({
  1077. name:'Create db',
  1078. //predicate: (sqlite3)=>
  1079. test: function(sqlite3){
  1080. const dbFile = '/tester1.db';
  1081. sqlite3.util.sqlite3__wasm_vfs_unlink(0, dbFile);
  1082. const db = this.db = new sqlite3.oo1.DB(dbFile, 0 ? 'ct' : 'c');
  1083. db.onclose = {
  1084. disposeAfter: [],
  1085. disposeBefore: [
  1086. (db)=>{
  1087. //console.debug("db.onclose.before dropping modules");
  1088. //sqlite3.capi.sqlite3_drop_modules(db.pointer, 0);
  1089. }
  1090. ],
  1091. before: function(db){
  1092. while(this.disposeBefore.length){
  1093. const v = this.disposeBefore.shift();
  1094. console.debug("db.onclose.before cleaning up:",v);
  1095. if(wasm.isPtr(v)) wasm.dealloc(v);
  1096. else if(v instanceof sqlite3.StructBinder.StructType){
  1097. v.dispose();
  1098. }else if(v instanceof Function){
  1099. try{ v(db) } catch(e){
  1100. console.warn("beforeDispose() callback threw:",e);
  1101. }
  1102. }
  1103. }
  1104. },
  1105. after: function(){
  1106. while(this.disposeAfter.length){
  1107. const v = this.disposeAfter.shift();
  1108. console.debug("db.onclose.after cleaning up:",v);
  1109. if(wasm.isPtr(v)) wasm.dealloc(v);
  1110. else if(v instanceof sqlite3.StructBinder.StructType){
  1111. v.dispose();
  1112. }else if(v instanceof Function){
  1113. try{v()} catch(e){/*ignored*/}
  1114. }
  1115. }
  1116. }
  1117. };
  1118. T.assert(wasm.isPtr(db.pointer))
  1119. .mustThrowMatching(()=>db.pointer=1, /read-only/)
  1120. .assert(0===sqlite3.capi.sqlite3_extended_result_codes(db.pointer,1))
  1121. .assert('main'===db.dbName(0))
  1122. .assert('string' === typeof db.dbVfsName())
  1123. .assert(db.pointer === wasm.xWrap.testConvertArg('sqlite3*',db));
  1124. // Custom db error message handling via sqlite3_prepare_v2/v3()
  1125. let rc = capi.sqlite3_prepare_v3(db.pointer, {/*invalid*/}, -1, 0, null, null);
  1126. T.assert(capi.SQLITE_MISUSE === rc)
  1127. .assert(0 === capi.sqlite3_errmsg(db.pointer).indexOf("Invalid SQL"))
  1128. .assert(dbFile === db.dbFilename())
  1129. .assert(!db.dbFilename('nope'));
  1130. //Sanity check DB.checkRc()...
  1131. let ex;
  1132. try{db.checkRc(rc)}
  1133. catch(e){ex = e}
  1134. T.assert(ex instanceof sqlite3.SQLite3Error)
  1135. .assert(capi.SQLITE_MISUSE===ex.resultCode)
  1136. .assert(0===ex.message.indexOf("SQLITE_MISUSE: sqlite3 result code"))
  1137. .assert(ex.message.indexOf("Invalid SQL")>0);
  1138. T.assert(db === db.checkRc(0))
  1139. .assert(db === sqlite3.oo1.DB.checkRc(db,0))
  1140. .assert(null === sqlite3.oo1.DB.checkRc(null,0));
  1141. this.progressHandlerCount = 0;
  1142. if( wasm.compileOptionUsed('OMIT_PROGRESS_CALLBACK') ){
  1143. T.assert( !capi.sqlite3_progress_handler );
  1144. }else{
  1145. T.assert( !!capi.sqlite3_progress_handler );
  1146. capi.sqlite3_progress_handler(db, 5, (p)=>{
  1147. ++this.progressHandlerCount;
  1148. return 0;
  1149. }, 0);
  1150. }
  1151. }
  1152. })
  1153. ////////////////////////////////////////////////////////////////////
  1154. .t('sqlite3_db_config() and sqlite3_db_status()', function(sqlite3){
  1155. let rc = capi.sqlite3_db_config(this.db, capi.SQLITE_DBCONFIG_LEGACY_ALTER_TABLE, 0, 0);
  1156. T.assert(0===rc);
  1157. rc = capi.sqlite3_db_config(this.db, capi.SQLITE_DBCONFIG_MAX+1, 0);
  1158. T.assert(capi.SQLITE_MISUSE === rc);
  1159. rc = capi.sqlite3_db_config(this.db, capi.SQLITE_DBCONFIG_MAINDBNAME, "main");
  1160. T.assert(0 === rc);
  1161. const stack = wasm.pstack.pointer;
  1162. try {
  1163. const [pCur, pHi] = wasm.pstack.allocChunks(2,'i64');
  1164. wasm.poke32([pCur, pHi], 0);
  1165. let [vCur, vHi] = wasm.peek32(pCur, pHi);
  1166. T.assert(0===vCur).assert(0===vHi);
  1167. rc = capi.sqlite3_status(capi.SQLITE_STATUS_MEMORY_USED,
  1168. pCur, pHi, 0);
  1169. [vCur, vHi] = wasm.peek32(pCur, pHi);
  1170. //console.warn("i32 vCur,vHi",vCur,vHi);
  1171. T.assert(0 === rc).assert(vCur > 0).assert(vHi >= vCur);
  1172. if(wasm.bigIntEnabled){
  1173. // Again in 64-bit. Recall that pCur and pHi are allocated
  1174. // large enough to account for this re-use.
  1175. wasm.poke64([pCur, pHi], 0);
  1176. rc = capi.sqlite3_status64(capi.SQLITE_STATUS_MEMORY_USED,
  1177. pCur, pHi, 0);
  1178. [vCur, vHi] = wasm.peek64([pCur, pHi]);
  1179. //console.warn("i64 vCur,vHi",vCur,vHi);
  1180. T.assert(0 === rc).assert(vCur > 0).assert(vHi >= vCur);
  1181. }
  1182. }finally{
  1183. wasm.pstack.restore(stack);
  1184. }
  1185. })
  1186. ////////////////////////////////////////////////////////////////////
  1187. .t('DB.Stmt', function(sqlite3){
  1188. let st = this.db.prepare(
  1189. new TextEncoder('utf-8').encode("select 3 as a")
  1190. );
  1191. let rc;
  1192. try {
  1193. T.assert(wasm.isPtr(st.pointer))
  1194. .mustThrowMatching(()=>st.pointer=1, /read-only/)
  1195. .assert(1===this.db.openStatementCount())
  1196. .assert(
  1197. capi.sqlite3_stmt_status(
  1198. st, capi.SQLITE_STMTSTATUS_RUN, 0
  1199. ) === 0)
  1200. .assert(!st._mayGet)
  1201. .assert('a' === st.getColumnName(0))
  1202. .mustThrowMatching(()=>st.columnCount=2,
  1203. /columnCount property is read-only/)
  1204. .assert(1===st.columnCount)
  1205. .assert(0===st.parameterCount)
  1206. .mustThrow(()=>st.bind(1,null))
  1207. .assert(true===st.step())
  1208. .assert(3 === st.get(0))
  1209. .mustThrow(()=>st.get(1))
  1210. .mustThrow(()=>st.get(0,~capi.SQLITE_INTEGER))
  1211. .assert(3 === st.get(0,capi.SQLITE_INTEGER))
  1212. .assert(3 === st.getInt(0))
  1213. .assert('3' === st.get(0,capi.SQLITE_TEXT))
  1214. .assert('3' === st.getString(0))
  1215. .assert(3.0 === st.get(0,capi.SQLITE_FLOAT))
  1216. .assert(3.0 === st.getFloat(0))
  1217. .assert(3 === st.get({}).a)
  1218. .assert(3 === st.get([])[0])
  1219. .assert(3 === st.getJSON(0))
  1220. .assert(st.get(0,capi.SQLITE_BLOB) instanceof Uint8Array)
  1221. .assert(1===st.get(0,capi.SQLITE_BLOB).length)
  1222. .assert(st.getBlob(0) instanceof Uint8Array)
  1223. .assert('3'.charCodeAt(0) === st.getBlob(0)[0])
  1224. .assert(st._mayGet)
  1225. .assert(false===st.step())
  1226. .assert(!st._mayGet)
  1227. .assert(
  1228. capi.sqlite3_stmt_status(
  1229. st, capi.SQLITE_STMTSTATUS_RUN, 0
  1230. ) > 0);
  1231. T.assert(this.progressHandlerCount>0
  1232. || wasm.compileOptionUsed('OMIT_PROGRESS_CALLBACK'),
  1233. "Expecting progress callback.").
  1234. assert(0===capi.sqlite3_strglob("*.txt", "foo.txt")).
  1235. assert(0!==capi.sqlite3_strglob("*.txt", "foo.xtx")).
  1236. assert(0===capi.sqlite3_strlike("%.txt", "foo.txt", 0)).
  1237. assert(0!==capi.sqlite3_strlike("%.txt", "foo.xtx", 0));
  1238. }finally{
  1239. rc = st.finalize();
  1240. }
  1241. T.assert(!st.pointer)
  1242. .assert(0===this.db.openStatementCount())
  1243. .assert(0===rc);
  1244. T.mustThrowMatching(()=>new sqlite3.oo1.Stmt("hi"), function(err){
  1245. return (err instanceof sqlite3.SQLite3Error)
  1246. && capi.SQLITE_MISUSE === err.resultCode
  1247. && 0 < err.message.indexOf("Do not call the Stmt constructor directly.")
  1248. });
  1249. })
  1250. ////////////////////////////////////////////////////////////////////////
  1251. .t('sqlite3_js_...()', function(){
  1252. const db = this.db;
  1253. if(1){
  1254. const vfsList = capi.sqlite3_js_vfs_list();
  1255. T.assert(vfsList.length>1);
  1256. wasm.scopedAllocCall(()=>{
  1257. const vfsArg = (v)=>wasm.xWrap.testConvertArg('sqlite3_vfs*',v);
  1258. for(const v of vfsList){
  1259. T.assert('string' === typeof v);
  1260. const pVfs = capi.sqlite3_vfs_find(v);
  1261. T.assert(wasm.isPtr(pVfs))
  1262. .assert(pVfs===vfsArg(v));
  1263. const vfs = new capi.sqlite3_vfs(pVfs);
  1264. try { T.assert(vfsArg(vfs)===pVfs) }
  1265. finally{ vfs.dispose() }
  1266. }
  1267. });
  1268. }
  1269. /**
  1270. Trivia: the magic db name ":memory:" does not actually use the
  1271. "memdb" VFS unless "memdb" is _explicitly_ provided as the VFS
  1272. name. Instead, it uses the default VFS with an in-memory btree.
  1273. Thus this.db's VFS may not be memdb even though it's an in-memory
  1274. db.
  1275. */
  1276. const pVfsMem = capi.sqlite3_vfs_find('memdb'),
  1277. pVfsDflt = capi.sqlite3_vfs_find(0),
  1278. pVfsDb = capi.sqlite3_js_db_vfs(db.pointer);
  1279. T.assert(pVfsMem > 0)
  1280. .assert(pVfsDflt > 0)
  1281. .assert(pVfsDb > 0)
  1282. .assert(pVfsMem !== pVfsDflt
  1283. /* memdb lives on top of the default vfs */)
  1284. .assert(pVfsDb === pVfsDflt || pVfsdb === pVfsMem)
  1285. ;
  1286. /*const vMem = new capi.sqlite3_vfs(pVfsMem),
  1287. vDflt = new capi.sqlite3_vfs(pVfsDflt),
  1288. vDb = new capi.sqlite3_vfs(pVfsDb);*/
  1289. const duv = capi.sqlite3_js_db_uses_vfs;
  1290. T.assert(pVfsDflt === duv(db.pointer, 0)
  1291. || pVfsMem === duv(db.pointer,0))
  1292. .assert(!duv(db.pointer, "foo"))
  1293. ;
  1294. }/*sqlite3_js_...()*/)
  1295. ////////////////////////////////////////////////////////////////////
  1296. .t('Table t', function(sqlite3){
  1297. const db = this.db;
  1298. let list = [];
  1299. this.progressHandlerCount = 0;
  1300. let rc = db.exec({
  1301. sql:['CREATE TABLE t(a,b);',
  1302. // ^^^ using TEMP TABLE breaks the db export test
  1303. "INSERT INTO t(a,b) VALUES(1,2),(3,4),",
  1304. "(?,?)"/*intentionally missing semicolon to test for
  1305. off-by-one bug in string-to-WASM conversion*/],
  1306. saveSql: list,
  1307. bind: [5,6]
  1308. });
  1309. //debug("Exec'd SQL:", list);
  1310. T.assert(rc === db)
  1311. .assert(2 === list.length)
  1312. .assert('string'===typeof list[1])
  1313. .assert(3===db.changes())
  1314. .assert(this.progressHandlerCount > 0
  1315. || wasm.compileOptionUsed('OMIT_PROGRESS_CALLBACK'),
  1316. "Expecting progress callback.")
  1317. if(wasm.bigIntEnabled){
  1318. T.assert(3n===db.changes(false,true));
  1319. }
  1320. rc = db.exec({
  1321. sql: "INSERT INTO t(a,b) values('blob',X'6869') RETURNING 13",
  1322. rowMode: 0
  1323. });
  1324. T.assert(Array.isArray(rc))
  1325. .assert(1===rc.length)
  1326. .assert(13 === rc[0])
  1327. .assert(1===db.changes());
  1328. let vals = db.selectValues('select a from t order by a limit 2');
  1329. T.assert( 2 === vals.length )
  1330. .assert( 1===vals[0] && 3===vals[1] );
  1331. vals = db.selectValues('select a from t order by a limit $L',
  1332. {$L:2}, capi.SQLITE_TEXT);
  1333. T.assert( 2 === vals.length )
  1334. .assert( '1'===vals[0] && '3'===vals[1] );
  1335. vals = undefined;
  1336. let blob = db.selectValue("select b from t where a='blob'");
  1337. T.assert(blob instanceof Uint8Array).
  1338. assert(0x68===blob[0] && 0x69===blob[1]);
  1339. blob = null;
  1340. let counter = 0, colNames = [];
  1341. list.length = 0;
  1342. db.exec(new TextEncoder('utf-8').encode("SELECT a a, b b FROM t"),{
  1343. rowMode: 'object',
  1344. resultRows: list,
  1345. columnNames: colNames,
  1346. _myState: 3 /* Accessible from the callback */,
  1347. callback: function(row,stmt){
  1348. ++counter;
  1349. T.assert(
  1350. 3 === this._myState
  1351. /* Recall that "this" is the options object. */
  1352. ).assert(
  1353. this.columnNames===colNames
  1354. ).assert(
  1355. this.columnNames[0]==='a' && this.columnNames[1]==='b'
  1356. ).assert(
  1357. (row.a%2 && row.a<6) || 'blob'===row.a
  1358. );
  1359. }
  1360. });
  1361. T.assert(2 === colNames.length)
  1362. .assert('a' === colNames[0])
  1363. .assert(4 === counter)
  1364. .assert(4 === list.length);
  1365. colNames = [];
  1366. db.exec({
  1367. /* Ensure that columnNames is populated for empty result sets. */
  1368. sql: "SELECT a a, b B FROM t WHERE 0",
  1369. columnNames: colNames
  1370. });
  1371. T.assert(2===colNames.length)
  1372. .assert('a'===colNames[0] && 'B'===colNames[1]);
  1373. list.length = 0;
  1374. db.exec("SELECT a a, b b FROM t",{
  1375. rowMode: 'array',
  1376. callback: function(row,stmt){
  1377. ++counter;
  1378. T.assert(Array.isArray(row))
  1379. .assert((0===row[1]%2 && row[1]<7)
  1380. || (row[1] instanceof Uint8Array));
  1381. }
  1382. });
  1383. T.assert(8 === counter);
  1384. T.assert(Number.MIN_SAFE_INTEGER ===
  1385. db.selectValue("SELECT "+Number.MIN_SAFE_INTEGER)).
  1386. assert(Number.MAX_SAFE_INTEGER ===
  1387. db.selectValue("SELECT "+Number.MAX_SAFE_INTEGER));
  1388. counter = 0;
  1389. let rv = db.exec({
  1390. sql: "SELECT a FROM t",
  1391. callback: ()=>(1===++counter),
  1392. });
  1393. T.assert(db === rv)
  1394. .assert(2===counter,
  1395. "Expecting exec step() loop to stop if callback returns false.");
  1396. /** If exec() is passed neither callback nor returnValue but
  1397. is passed an explicit rowMode then the default returnValue
  1398. is the whole result set, as if an empty resultRows option
  1399. had been passed. */
  1400. rv = db.exec({
  1401. sql: "SELECT -1 UNION ALL SELECT -2 UNION ALL SELECT -3 ORDER BY 1 DESC",
  1402. rowMode: 0
  1403. });
  1404. T.assert(Array.isArray(rv)).assert(3===rv.length)
  1405. .assert(-1===rv[0]).assert(-3===rv[2]);
  1406. rv = db.exec("SELECT 1 WHERE 0",{rowMode: 0});
  1407. T.assert(Array.isArray(rv)).assert(0===rv.length);
  1408. if(wasm.bigIntEnabled && haveWasmCTests()){
  1409. const mI = wasm.xCall('sqlite3__wasm_test_int64_max');
  1410. const b = BigInt(Number.MAX_SAFE_INTEGER * 2);
  1411. T.assert(b === db.selectValue("SELECT "+b)).
  1412. assert(b === db.selectValue("SELECT ?", b)).
  1413. assert(mI == db.selectValue("SELECT $x", {$x:mI}));
  1414. }else{
  1415. /* Curiously, the JS spec seems to be off by one with the definitions
  1416. of MIN/MAX_SAFE_INTEGER:
  1417. https://github.com/emscripten-core/emscripten/issues/17391 */
  1418. T.mustThrow(()=>db.selectValue("SELECT "+(Number.MAX_SAFE_INTEGER+1))).
  1419. mustThrow(()=>db.selectValue("SELECT "+(Number.MIN_SAFE_INTEGER-1)));
  1420. }
  1421. let st = db.prepare("update t set b=:b where a='blob'");
  1422. try {
  1423. T.assert(0===st.columnCount)
  1424. .assert( false===st.isReadOnly() );
  1425. const ndx = st.getParamIndex(':b');
  1426. T.assert(1===ndx);
  1427. st.bindAsBlob(ndx, "ima blob")
  1428. /*step() skipped intentionally*/.reset(true);
  1429. } finally {
  1430. T.assert(0===st.finalize())
  1431. .assert(undefined===st.finalize());
  1432. }
  1433. try {
  1434. db.prepare("/*empty SQL*/");
  1435. toss("Must not be reached.");
  1436. }catch(e){
  1437. T.assert(e instanceof sqlite3.SQLite3Error)
  1438. .assert(0==e.message.indexOf('Cannot prepare empty'));
  1439. }
  1440. counter = 0;
  1441. db.exec({
  1442. // Check for https://sqlite.org/forum/forumpost/895425b49a
  1443. sql: "pragma table_info('t')",
  1444. rowMode: 'object',
  1445. callback: function(row){
  1446. ++counter;
  1447. T.assert(row.name==='a' || row.name==='b');
  1448. }
  1449. });
  1450. T.assert(2===counter);
  1451. })/*setup table T*/
  1452. ////////////////////////////////////////////////////////////////////
  1453. .t({
  1454. name: "sqlite3_set_authorizer()",
  1455. predicate: ()=>!!wasm.exports.sqlite3_set_authorizer || "Missing sqlite3_set_authorizer()",
  1456. test:function(sqlite3){
  1457. T.assert(capi.SQLITE_IGNORE>0)
  1458. .assert(capi.SQLITE_DENY>0);
  1459. const db = this.db;
  1460. const ssa = capi.sqlite3_set_authorizer;
  1461. const n = db.selectValue('select count(*) from t');
  1462. T.assert(n>0);
  1463. let authCount = 0;
  1464. let rc = ssa(db, function(pV, iCode, s0, s1, s2, s3){
  1465. ++authCount;
  1466. return capi.SQLITE_IGNORE;
  1467. }, 0);
  1468. T.assert(0===rc)
  1469. .assert(
  1470. undefined === db.selectValue('select count(*) from t')
  1471. /* Note that the count() never runs, so we get undefined
  1472. instead of 0. */
  1473. )
  1474. .assert(authCount>0);
  1475. authCount = 0;
  1476. db.exec("update t set a=-9999");
  1477. T.assert(authCount>0);
  1478. /* Reminder: we don't use DELETE because, from the C API docs:
  1479. "If the action code is [SQLITE_DELETE] and the callback
  1480. returns [SQLITE_IGNORE] then the [DELETE] operation proceeds
  1481. but the [truncate optimization] is disabled and all rows are
  1482. deleted individually."
  1483. */
  1484. rc = ssa(db, null, 0);
  1485. authCount = 0;
  1486. T.assert(-9999 != db.selectValue('select a from t'))
  1487. .assert(0===authCount);
  1488. rc = ssa(db, function(pV, iCode, s0, s1, s2, s3){
  1489. ++authCount;
  1490. return capi.SQLITE_DENY;
  1491. }, 0);
  1492. T.assert(0===rc);
  1493. let err;
  1494. try{ db.exec("select 1 from t") }
  1495. catch(e){ err = e }
  1496. T.assert(err instanceof sqlite3.SQLite3Error)
  1497. .assert(err.message.indexOf('not authorized'>0))
  1498. .assert(1===authCount);
  1499. authCount = 0;
  1500. rc = ssa(db, function(...args){
  1501. ++authCount;
  1502. return capi.SQLITE_OK;
  1503. }, 0);
  1504. T.assert(0===rc);
  1505. T.assert(n === db.selectValue('select count(*) from t'))
  1506. .assert(authCount>0);
  1507. authCount = 0;
  1508. rc = ssa(db, function(pV, iCode, s0, s1, s2, s3){
  1509. ++authCount;
  1510. throw new Error("Testing catching of authorizer.");
  1511. }, 0);
  1512. T.assert(0===rc);
  1513. authCount = 0;
  1514. err = undefined;
  1515. try{ db.exec("select 1 from t") }
  1516. catch(e){err = e}
  1517. T.assert(err instanceof Error)
  1518. .assert(err.message.indexOf('not authorized')>0)
  1519. /* Note that the thrown message is trumped/overwritten
  1520. by the authorizer process. */
  1521. .assert(1===authCount);
  1522. rc = ssa(db, 0, 0);
  1523. authCount = 0;
  1524. T.assert(0===rc);
  1525. T.assert(n === db.selectValue('select count(*) from t'))
  1526. .assert(0===authCount);
  1527. }
  1528. })/*sqlite3_set_authorizer()*/
  1529. ////////////////////////////////////////////////////////////////////////
  1530. .t("sqlite3_table_column_metadata()", function(sqlite3){
  1531. const stack = wasm.pstack.pointer;
  1532. try{
  1533. const [pzDT, pzColl, pNotNull, pPK, pAuto] =
  1534. wasm.pstack.allocPtr(5);
  1535. const rc = capi.sqlite3_table_column_metadata(
  1536. this.db, "main", "t", "rowid",
  1537. pzDT, pzColl, pNotNull, pPK, pAuto
  1538. );
  1539. T.assert(0===rc)
  1540. .assert("INTEGER"===wasm.cstrToJs(wasm.peekPtr(pzDT)))
  1541. .assert("BINARY"===wasm.cstrToJs(wasm.peekPtr(pzColl)))
  1542. .assert(0===wasm.peek32(pNotNull))
  1543. .assert(1===wasm.peek32(pPK))
  1544. .assert(0===wasm.peek32(pAuto))
  1545. }finally{
  1546. wasm.pstack.restore(stack);
  1547. }
  1548. })
  1549. ////////////////////////////////////////////////////////////////////////
  1550. .t('selectArray/Object()', function(sqlite3){
  1551. const db = this.db;
  1552. let rc = db.selectArray('select a, b from t where a=?', 5);
  1553. T.assert(Array.isArray(rc))
  1554. .assert(2===rc.length)
  1555. .assert(5===rc[0] && 6===rc[1]);
  1556. rc = db.selectArray('select a, b from t where b=-1');
  1557. T.assert(undefined === rc);
  1558. rc = db.selectObject('select a A, b b from t where b=?', 6);
  1559. T.assert(rc && 'object'===typeof rc)
  1560. .assert(5===rc.A)
  1561. .assert(6===rc.b);
  1562. rc = db.selectArray('select a, b from t where b=-1');
  1563. T.assert(undefined === rc);
  1564. })
  1565. ////////////////////////////////////////////////////////////////////////
  1566. .t('selectArrays/Objects()', function(sqlite3){
  1567. const db = this.db;
  1568. const sql = 'select a, b from t where a=? or b=? order by a';
  1569. let rc = db.selectArrays(sql, [1, 4]);
  1570. T.assert(Array.isArray(rc))
  1571. .assert(2===rc.length)
  1572. .assert(2===rc[0].length)
  1573. .assert(1===rc[0][0])
  1574. .assert(2===rc[0][1])
  1575. .assert(3===rc[1][0])
  1576. .assert(4===rc[1][1])
  1577. rc = db.selectArrays(sql, [99,99]);
  1578. T.assert(Array.isArray(rc)).assert(0===rc.length);
  1579. rc = db.selectObjects(sql, [1,4]);
  1580. T.assert(Array.isArray(rc))
  1581. .assert(2===rc.length)
  1582. .assert('object' === typeof rc[1])
  1583. .assert(1===rc[0].a)
  1584. .assert(2===rc[0].b)
  1585. .assert(3===rc[1].a)
  1586. .assert(4===rc[1].b);
  1587. })
  1588. ////////////////////////////////////////////////////////////////////////
  1589. .t('selectArray/Object/Values() via INSERT/UPDATE...RETURNING', function(sqlite3){
  1590. let rc = this.db.selectObject("INSERT INTO t(a,b) VALUES(83,84) RETURNING a as AA");
  1591. T.assert(83===rc.AA);
  1592. rc = this.db.selectArray("UPDATE T set a=85 WHERE a=83 RETURNING b as BB");
  1593. T.assert(Array.isArray(rc)).assert(84===rc[0]);
  1594. //log("select * from t:",this.db.selectObjects("select * from t order by a"));
  1595. rc = this.db.selectValues("UPDATE T set a=a*1 RETURNING a");
  1596. T.assert(Array.isArray(rc))
  1597. .assert(5 === rc.length)
  1598. .assert('number'===typeof rc[0])
  1599. .assert(rc[0]|0 === rc[0] /* is small integer */);
  1600. })
  1601. ////////////////////////////////////////////////////////////////////////
  1602. .t({
  1603. name: 'sqlite3_js_db_export()',
  1604. predicate: ()=>true,
  1605. test: function(sqlite3){
  1606. const db = this.db;
  1607. const xp = capi.sqlite3_js_db_export(db.pointer);
  1608. T.assert(xp instanceof Uint8Array)
  1609. .assert(xp.byteLength>0)
  1610. .assert(0 === xp.byteLength % 512);
  1611. this.dbExport = xp;
  1612. }
  1613. }/*sqlite3_js_db_export()*/)
  1614. .t({
  1615. name: 'sqlite3_js_posix_create_file()',
  1616. predicate: ()=>true,
  1617. test: function(sqlite3){
  1618. const db = this.db;
  1619. const filename = "sqlite3_js_posix_create_file.db";
  1620. capi.sqlite3_js_posix_create_file(filename, this.dbExport);
  1621. delete this.dbExport;
  1622. const db2 = new sqlite3.oo1.DB(filename,'r');
  1623. try {
  1624. const sql = "select count(*) from t";
  1625. const n = db.selectValue(sql);
  1626. T.assert(n>0 && db2.selectValue(sql) === n);
  1627. }finally{
  1628. db2.close();
  1629. sqlite3.util.sqlite3__wasm_vfs_unlink(0, filename);
  1630. }
  1631. }
  1632. }/*sqlite3_js_posix_create_file()*/)
  1633. ////////////////////////////////////////////////////////////////////
  1634. .t({
  1635. name:'Scalar UDFs',
  1636. test: function(sqlite3){
  1637. const db = this.db;
  1638. db.createFunction("foo",(pCx,a,b)=>a+b);
  1639. T.assert(7===db.selectValue("select foo(3,4)")).
  1640. assert(5===db.selectValue("select foo(3,?)",2)).
  1641. assert(5===db.selectValue("select foo(?,?2)",[1,4])).
  1642. assert(5===db.selectValue("select foo($a,$b)",{$a:0,$b:5}));
  1643. db.createFunction("bar", {
  1644. arity: -1,
  1645. xFunc: (pCx,...args)=>{
  1646. T.assert(db.pointer === capi.sqlite3_context_db_handle(pCx));
  1647. let rc = 0;
  1648. for(const v of args) rc += v;
  1649. return rc;
  1650. }
  1651. }).createFunction({
  1652. name: "asis",
  1653. xFunc: (pCx,arg)=>arg
  1654. });
  1655. T.assert(0===db.selectValue("select bar()")).
  1656. assert(1===db.selectValue("select bar(1)")).
  1657. assert(3===db.selectValue("select bar(1,2)")).
  1658. assert(-1===db.selectValue("select bar(1,2,-4)")).
  1659. assert('hi' === db.selectValue("select asis('hi')")).
  1660. assert('hi' === db.selectValue("select ?",'hi')).
  1661. assert(null === db.selectValue("select null")).
  1662. assert(null === db.selectValue("select asis(null)")).
  1663. assert(1 === db.selectValue("select ?",1)).
  1664. assert(2 === db.selectValue("select ?",[2])).
  1665. assert(3 === db.selectValue("select $a",{$a:3})).
  1666. assert(T.eqApprox(3.1,db.selectValue("select 3.0 + 0.1"))).
  1667. assert(T.eqApprox(1.3,db.selectValue("select asis(1 + 0.3)")));
  1668. let blobArg = new Uint8Array([0x68, 0x69]);
  1669. let blobRc = db.selectValue(
  1670. "select asis(?1)",
  1671. blobArg.buffer/*confirm that ArrayBuffer is handled as a Uint8Array*/
  1672. );
  1673. T.assert(blobRc instanceof Uint8Array).
  1674. assert(2 === blobRc.length).
  1675. assert(0x68==blobRc[0] && 0x69==blobRc[1]);
  1676. blobRc = db.selectValue("select asis(X'6869')");
  1677. T.assert(blobRc instanceof Uint8Array).
  1678. assert(2 === blobRc.length).
  1679. assert(0x68==blobRc[0] && 0x69==blobRc[1]);
  1680. blobArg = new Int8Array([0x68, 0x69]);
  1681. //debug("blobArg=",blobArg);
  1682. blobRc = db.selectValue("select asis(?1)", blobArg);
  1683. T.assert(blobRc instanceof Uint8Array).
  1684. assert(2 === blobRc.length);
  1685. //debug("blobRc=",blobRc);
  1686. T.assert(0x68==blobRc[0] && 0x69==blobRc[1]);
  1687. let rc = sqlite3.capi.sqlite3_create_function_v2(
  1688. this.db, "foo", 0, -1, 0, 0, 0, 0, 0
  1689. );
  1690. T.assert(
  1691. sqlite3.capi.SQLITE_FORMAT === rc,
  1692. "For invalid eTextRep argument."
  1693. );
  1694. rc = sqlite3.capi.sqlite3_create_function_v2(this.db, "foo", 0);
  1695. T.assert(
  1696. sqlite3.capi.SQLITE_MISUSE === rc,
  1697. "For invalid arg count."
  1698. );
  1699. /* Confirm that we can map and unmap the same function with
  1700. multiple arities... */
  1701. const fCounts = [0,0];
  1702. const fArityCheck = function(pCx){
  1703. return ++fCounts[arguments.length-1];
  1704. };
  1705. //wasm.xWrap.FuncPtrAdapter.debugFuncInstall = true;
  1706. rc = capi.sqlite3_create_function_v2(
  1707. db, "nary", 0, capi.SQLITE_UTF8, 0, fArityCheck, 0, 0, 0
  1708. );
  1709. T.assert( 0===rc );
  1710. rc = capi.sqlite3_create_function_v2(
  1711. db, "nary", 1, capi.SQLITE_UTF8, 0, fArityCheck, 0, 0, 0
  1712. );
  1713. T.assert( 0===rc );
  1714. const sqlFArity0 = "select nary()";
  1715. const sqlFArity1 = "select nary(1)";
  1716. T.assert( 1 === db.selectValue(sqlFArity0) )
  1717. .assert( 1 === fCounts[0] ).assert( 0 === fCounts[1] );
  1718. T.assert( 1 === db.selectValue(sqlFArity1) )
  1719. .assert( 1 === fCounts[0] ).assert( 1 === fCounts[1] );
  1720. capi.sqlite3_create_function_v2(
  1721. db, "nary", 0, capi.SQLITE_UTF8, 0, 0, 0, 0, 0
  1722. );
  1723. T.mustThrowMatching((()=>db.selectValue(sqlFArity0)),
  1724. (e)=>((e instanceof sqlite3.SQLite3Error)
  1725. && e.message.indexOf("wrong number of arguments")>0),
  1726. "0-arity variant was uninstalled.");
  1727. T.assert( 2 === db.selectValue(sqlFArity1) )
  1728. .assert( 1 === fCounts[0] ).assert( 2 === fCounts[1] );
  1729. capi.sqlite3_create_function_v2(
  1730. db, "nary", 1, capi.SQLITE_UTF8, 0, 0, 0, 0, 0
  1731. );
  1732. T.mustThrowMatching((()=>db.selectValue(sqlFArity1)),
  1733. (e)=>((e instanceof sqlite3.SQLite3Error)
  1734. && e.message.indexOf("no such function")>0),
  1735. "1-arity variant was uninstalled.");
  1736. //wasm.xWrap.FuncPtrAdapter.debugFuncInstall = false;
  1737. }
  1738. })
  1739. ////////////////////////////////////////////////////////////////////
  1740. .t({
  1741. name: 'Aggregate UDFs',
  1742. //predicate: ()=>false,
  1743. test: function(sqlite3){
  1744. const db = this.db;
  1745. const sjac = capi.sqlite3_js_aggregate_context;
  1746. db.createFunction({
  1747. name: 'summer',
  1748. xStep: (pCtx, n)=>{
  1749. const ac = sjac(pCtx, 4);
  1750. wasm.poke32(ac, wasm.peek32(ac) + Number(n));
  1751. },
  1752. xFinal: (pCtx)=>{
  1753. const ac = sjac(pCtx, 0);
  1754. return ac ? wasm.peek32(ac) : 0;
  1755. }
  1756. });
  1757. let v = db.selectValue([
  1758. "with cte(v) as (",
  1759. "select 3 union all select 5 union all select 7",
  1760. ") select summer(v), summer(v+1) from cte"
  1761. /* ------------------^^^^^^^^^^^ ensures that we're handling
  1762. sqlite3_aggregate_context() properly. */
  1763. ]);
  1764. T.assert(15===v);
  1765. T.mustThrowMatching(()=>db.selectValue("select summer(1,2)"),
  1766. /wrong number of arguments/);
  1767. db.createFunction({
  1768. name: 'summerN',
  1769. arity: -1,
  1770. xStep: (pCtx, ...args)=>{
  1771. const ac = sjac(pCtx, 4);
  1772. let sum = wasm.peek32(ac);
  1773. for(const v of args) sum += Number(v);
  1774. wasm.poke32(ac, sum);
  1775. },
  1776. xFinal: (pCtx)=>{
  1777. const ac = sjac(pCtx, 0);
  1778. capi.sqlite3_result_int( pCtx, ac ? wasm.peek32(ac) : 0 );
  1779. // xFinal() may either return its value directly or call
  1780. // sqlite3_result_xyz() and return undefined. Both are
  1781. // functionally equivalent.
  1782. }
  1783. });
  1784. T.assert(18===db.selectValue('select summerN(1,8,9), summerN(2,3,4)'));
  1785. T.mustThrowMatching(()=>{
  1786. db.createFunction('nope',{
  1787. xFunc: ()=>{}, xStep: ()=>{}
  1788. });
  1789. }, /scalar or aggregate\?/);
  1790. T.mustThrowMatching(()=>{
  1791. db.createFunction('nope',{xStep: ()=>{}});
  1792. }, /Missing xFinal/);
  1793. T.mustThrowMatching(()=>{
  1794. db.createFunction('nope',{xFinal: ()=>{}});
  1795. }, /Missing xStep/);
  1796. T.mustThrowMatching(()=>{
  1797. db.createFunction('nope',{});
  1798. }, /Missing function-type properties/);
  1799. T.mustThrowMatching(()=>{
  1800. db.createFunction('nope',{xFunc:()=>{}, xDestroy:'nope'});
  1801. }, /xDestroy property must be a function/);
  1802. T.mustThrowMatching(()=>{
  1803. db.createFunction('nope',{xFunc:()=>{}, pApp:'nope'});
  1804. }, /Invalid value for pApp/);
  1805. }
  1806. }/*aggregate UDFs*/)
  1807. ////////////////////////////////////////////////////////////////////////
  1808. .t({
  1809. name: 'Aggregate UDFs (64-bit)',
  1810. predicate: ()=>wasm.bigIntEnabled,
  1811. //predicate: ()=>false,
  1812. test: function(sqlite3){
  1813. const db = this.db;
  1814. const sjac = capi.sqlite3_js_aggregate_context;
  1815. db.createFunction({
  1816. name: 'summer64',
  1817. xStep: (pCtx, n)=>{
  1818. const ac = sjac(pCtx, 8);
  1819. wasm.poke64(ac, wasm.peek64(ac) + BigInt(n));
  1820. },
  1821. xFinal: (pCtx)=>{
  1822. const ac = sjac(pCtx, 0);
  1823. return ac ? wasm.peek64(ac) : 0n;
  1824. }
  1825. });
  1826. let v = db.selectValue([
  1827. "with cte(v) as (",
  1828. "select 9007199254740991 union all select 1 union all select 2",
  1829. ") select summer64(v), summer64(v+1) from cte"
  1830. ]);
  1831. T.assert(9007199254740994n===v);
  1832. }
  1833. }/*aggregate UDFs*/)
  1834. ////////////////////////////////////////////////////////////////////
  1835. .t({
  1836. name: 'Window UDFs',
  1837. predicate: (sqlite3)=>!!sqlite3.wasm.exports.sqlite3_create_window_function
  1838. /*!sqlite3.wasm.compileOptionUsed('OMIT_WINDOWFUNC')*/
  1839. || "Missing window functions",
  1840. test: function(){
  1841. /* Example window function, table, and results taken from:
  1842. https://sqlite.org/windowfunctions.html#udfwinfunc */
  1843. const db = this.db;
  1844. const sjac = (cx,n=4)=>capi.sqlite3_js_aggregate_context(cx,n);
  1845. const xValueFinal = (pCtx)=>{
  1846. const ac = sjac(pCtx, 0);
  1847. return ac ? wasm.peek32(ac) : 0;
  1848. };
  1849. const xStepInverse = (pCtx, n)=>{
  1850. const ac = sjac(pCtx);
  1851. wasm.poke32(ac, wasm.peek32(ac) + Number(n));
  1852. };
  1853. db.createFunction({
  1854. name: 'winsumint',
  1855. xStep: (pCtx, n)=>xStepInverse(pCtx, n),
  1856. xInverse: (pCtx, n)=>xStepInverse(pCtx, -n),
  1857. xFinal: xValueFinal,
  1858. xValue: xValueFinal
  1859. });
  1860. db.exec([
  1861. "CREATE TEMP TABLE twin(x, y); INSERT INTO twin VALUES",
  1862. "('a', 4),('b', 5),('c', 3),('d', 8),('e', 1)"
  1863. ]);
  1864. let rc = db.exec({
  1865. returnValue: 'resultRows',
  1866. sql:[
  1867. "SELECT x, winsumint(y) OVER (",
  1868. "ORDER BY x ROWS BETWEEN 1 PRECEDING AND 1 FOLLOWING",
  1869. ") AS sum_y ",
  1870. "FROM twin ORDER BY x;"
  1871. ]
  1872. });
  1873. T.assert(Array.isArray(rc))
  1874. .assert(5 === rc.length);
  1875. let count = 0;
  1876. for(const row of rc){
  1877. switch(++count){
  1878. case 1: T.assert('a'===row[0] && 9===row[1]); break;
  1879. case 2: T.assert('b'===row[0] && 12===row[1]); break;
  1880. case 3: T.assert('c'===row[0] && 16===row[1]); break;
  1881. case 4: T.assert('d'===row[0] && 12===row[1]); break;
  1882. case 5: T.assert('e'===row[0] && 9===row[1]); break;
  1883. default: toss("Too many rows to window function.");
  1884. }
  1885. }
  1886. const resultRows = [];
  1887. rc = db.exec({
  1888. resultRows,
  1889. returnValue: 'resultRows',
  1890. sql:[
  1891. "SELECT x, winsumint(y) OVER (",
  1892. "ORDER BY x ROWS BETWEEN 1 PRECEDING AND 1 FOLLOWING",
  1893. ") AS sum_y ",
  1894. "FROM twin ORDER BY x;"
  1895. ]
  1896. });
  1897. T.assert(rc === resultRows)
  1898. .assert(5 === rc.length);
  1899. rc = db.exec({
  1900. returnValue: 'saveSql',
  1901. sql: "select 1; select 2; -- empty\n; select 3"
  1902. });
  1903. T.assert(Array.isArray(rc))
  1904. .assert(3===rc.length)
  1905. .assert('select 1;' === rc[0])
  1906. .assert('select 2;' === rc[1])
  1907. .assert('-- empty\n; select 3' === rc[2]
  1908. /* Strange but true. */);
  1909. T.mustThrowMatching(()=>{
  1910. db.exec({sql:'', returnValue: 'nope'});
  1911. }, /^Invalid returnValue/);
  1912. db.exec("DROP TABLE twin");
  1913. }
  1914. }/*window UDFs*/)
  1915. ////////////////////////////////////////////////////////////////////
  1916. .t("ATTACH", function(){
  1917. const db = this.db;
  1918. const resultRows = [];
  1919. db.exec({
  1920. sql:new TextEncoder('utf-8').encode([
  1921. // ^^^ testing string-vs-typedarray handling in exec()
  1922. "attach 'session' as foo;",
  1923. "create table foo.bar(a);",
  1924. "insert into foo.bar(a) values(1),(2),(3);",
  1925. "select a from foo.bar order by a;"
  1926. ].join('')),
  1927. rowMode: 0,
  1928. resultRows
  1929. });
  1930. T.assert(3===resultRows.length)
  1931. .assert(2===resultRows[1]);
  1932. T.assert(2===db.selectValue('select a from foo.bar where a>1 order by a'));
  1933. /** Demonstrate the JS-simplified form of the sqlite3_exec() callback... */
  1934. let colCount = 0, rowCount = 0;
  1935. let rc = capi.sqlite3_exec(
  1936. db, "select a, a*2 from foo.bar", function(aVals, aNames){
  1937. //console.warn("execCallback(",arguments,")");
  1938. colCount = aVals.length;
  1939. ++rowCount;
  1940. T.assert(2===aVals.length)
  1941. .assert(2===aNames.length)
  1942. .assert(+(aVals[1]) === 2 * +(aVals[0]));
  1943. }, 0, 0
  1944. );
  1945. T.assert(0===rc).assert(3===rowCount).assert(2===colCount);
  1946. rc = capi.sqlite3_exec(
  1947. db.pointer, "select a from foo.bar", ()=>{
  1948. tossQuietly("Testing throwing from exec() callback.");
  1949. }, 0, 0
  1950. );
  1951. T.assert(capi.SQLITE_ABORT === rc);
  1952. /* Demonstrate how to get access to the "full" callback
  1953. signature, as opposed to the simplified JS-specific one... */
  1954. rowCount = colCount = 0;
  1955. const pCb = wasm.installFunction('i(pipp)', function(pVoid,nCols,aVals,aCols){
  1956. /* Tip: wasm.cArgvToJs() can be used to convert aVals and
  1957. aCols to arrays: const vals = wasm.cArgvToJs(nCols,
  1958. aVals); */
  1959. ++rowCount;
  1960. colCount = nCols;
  1961. T.assert(2 === nCols)
  1962. .assert(wasm.isPtr(pVoid))
  1963. .assert(wasm.isPtr(aVals))
  1964. .assert(wasm.isPtr(aCols))
  1965. .assert(+wasm.cstrToJs(wasm.peekPtr(aVals + wasm.ptrSizeof))
  1966. === 2 * +wasm.cstrToJs(wasm.peekPtr(aVals)));
  1967. return 0;
  1968. });
  1969. try {
  1970. T.assert(wasm.isPtr(pCb));
  1971. rc = capi.sqlite3_exec(
  1972. db, new TextEncoder('utf-8').encode("select a, a*2 from foo.bar"),
  1973. pCb, 0, 0
  1974. );
  1975. T.assert(0===rc)
  1976. .assert(3===rowCount)
  1977. .assert(2===colCount);
  1978. }finally{
  1979. wasm.uninstallFunction(pCb);
  1980. }
  1981. // Demonstrate that an OOM result does not propagate through sqlite3_exec()...
  1982. rc = capi.sqlite3_exec(
  1983. db, ["select a,"," a*2 from foo.bar"], (aVals, aNames)=>{
  1984. sqlite3.WasmAllocError.toss("just testing");
  1985. }, 0, 0
  1986. );
  1987. T.assert(capi.SQLITE_ABORT === rc);
  1988. db.exec("detach foo");
  1989. T.mustThrow(()=>db.exec("select * from foo.bar"),
  1990. "Because foo is no longer attached.");
  1991. })
  1992. ////////////////////////////////////////////////////////////////////
  1993. .t("Read-only", function(sqlite3){
  1994. T.assert( 0===capi.sqlite3_db_readonly(this.db, "main") );
  1995. const db = new sqlite3.oo1.DB('file://'+this.db.filename+'?mode=ro');
  1996. T.assert( 1===capi.sqlite3_db_readonly(db, "main") );
  1997. T.assert( -1===capi.sqlite3_db_readonly(db, "nope") );
  1998. db.close();
  1999. })
  2000. ////////////////////////////////////////////////////////////////////
  2001. .t({
  2002. name: 'C-side WASM tests',
  2003. predicate: ()=>(haveWasmCTests() || "Not compiled in."),
  2004. test: function(){
  2005. const w = wasm, db = this.db;
  2006. const stack = w.scopedAllocPush();
  2007. let ptrInt;
  2008. const origValue = 512;
  2009. try{
  2010. ptrInt = w.scopedAlloc(4);
  2011. w.poke32(ptrInt,origValue);
  2012. const cf = w.xGet('sqlite3__wasm_test_intptr');
  2013. const oldPtrInt = ptrInt;
  2014. T.assert(origValue === w.peek32(ptrInt));
  2015. const rc = cf(ptrInt);
  2016. T.assert(2*origValue === rc).
  2017. assert(rc === w.peek32(ptrInt)).
  2018. assert(oldPtrInt === ptrInt);
  2019. const pi64 = w.scopedAlloc(8)/*ptr to 64-bit integer*/;
  2020. const o64 = 0x010203040506/*>32-bit integer*/;
  2021. if(w.bigIntEnabled){
  2022. w.poke64(pi64, o64);
  2023. //log("pi64 =",pi64, "o64 = 0x",o64.toString(16), o64);
  2024. const v64 = ()=>w.peek64(pi64)
  2025. T.assert(v64() == o64);
  2026. //T.assert(o64 === w.peek64(pi64));
  2027. const cf64w = w.xGet('sqlite3__wasm_test_int64ptr');
  2028. cf64w(pi64);
  2029. T.assert(v64() == BigInt(2 * o64));
  2030. cf64w(pi64);
  2031. T.assert(v64() == BigInt(4 * o64));
  2032. const biTimes2 = w.xGet('sqlite3__wasm_test_int64_times2');
  2033. T.assert(BigInt(2 * o64) ===
  2034. biTimes2(BigInt(o64)/*explicit conv. required to avoid TypeError
  2035. in the call :/ */));
  2036. const pMin = w.scopedAlloc(16);
  2037. const pMax = pMin + 8;
  2038. const g64 = (p)=>w.peek64(p);
  2039. w.poke64([pMin, pMax], 0);
  2040. const minMaxI64 = [
  2041. w.xCall('sqlite3__wasm_test_int64_min'),
  2042. w.xCall('sqlite3__wasm_test_int64_max')
  2043. ];
  2044. T.assert(minMaxI64[0] < BigInt(Number.MIN_SAFE_INTEGER)).
  2045. assert(minMaxI64[1] > BigInt(Number.MAX_SAFE_INTEGER));
  2046. //log("int64_min/max() =",minMaxI64, typeof minMaxI64[0]);
  2047. w.xCall('sqlite3__wasm_test_int64_minmax', pMin, pMax);
  2048. T.assert(g64(pMin) === minMaxI64[0], "int64 mismatch").
  2049. assert(g64(pMax) === minMaxI64[1], "int64 mismatch");
  2050. //log("pMin",g64(pMin), "pMax",g64(pMax));
  2051. w.poke64(pMin, minMaxI64[0]);
  2052. T.assert(g64(pMin) === minMaxI64[0]).
  2053. assert(minMaxI64[0] === db.selectValue("select ?",g64(pMin))).
  2054. assert(minMaxI64[1] === db.selectValue("select ?",g64(pMax)));
  2055. const rxRange = /too big/;
  2056. T.mustThrowMatching(()=>{db.prepare("select ?").bind(minMaxI64[0] - BigInt(1))},
  2057. rxRange).
  2058. mustThrowMatching(()=>{db.prepare("select ?").bind(minMaxI64[1] + BigInt(1))},
  2059. (e)=>rxRange.test(e.message));
  2060. }else{
  2061. log("No BigInt support. Skipping related tests.");
  2062. log("\"The problem\" here is that we can manipulate, at the byte level,",
  2063. "heap memory to set 64-bit values, but we can't get those values",
  2064. "back into JS because of the lack of 64-bit integer support.");
  2065. }
  2066. }finally{
  2067. const x = w.scopedAlloc(1), y = w.scopedAlloc(1), z = w.scopedAlloc(1);
  2068. //log("x=",x,"y=",y,"z=",z); // just looking at the alignment
  2069. w.scopedAllocPop(stack);
  2070. }
  2071. }
  2072. }/* jaccwabyt-specific tests */)
  2073. ////////////////////////////////////////////////////////////////////////
  2074. .t({
  2075. name: 'virtual table #1: eponymous w/ manual exception handling',
  2076. predicate: (sqlite3)=>!!sqlite3.capi.sqlite3_vtab || "Missing vtab support",
  2077. test: function(sqlite3){
  2078. const VT = sqlite3.vtab;
  2079. const tmplCols = Object.assign(Object.create(null),{
  2080. A: 0, B: 1
  2081. });
  2082. /**
  2083. The vtab demonstrated here is a JS-ification of
  2084. ext/misc/templatevtab.c.
  2085. */
  2086. const tmplMod = new sqlite3.capi.sqlite3_module();
  2087. T.assert(0===tmplMod.$xUpdate);
  2088. tmplMod.setupModule({
  2089. catchExceptions: false,
  2090. methods: {
  2091. xConnect: function(pDb, pAux, argc, argv, ppVtab, pzErr){
  2092. try{
  2093. const args = wasm.cArgvToJs(argc, argv);
  2094. T.assert(args.length>=3)
  2095. .assert(args[0] === 'testvtab')
  2096. .assert(args[1] === 'main')
  2097. .assert(args[2] === 'testvtab');
  2098. //console.debug("xConnect() args =",args);
  2099. const rc = capi.sqlite3_declare_vtab(
  2100. pDb, "CREATE TABLE ignored(a,b)"
  2101. );
  2102. if(0===rc){
  2103. const t = VT.xVtab.create(ppVtab);
  2104. T.assert(t === VT.xVtab.get(wasm.peekPtr(ppVtab)));
  2105. }
  2106. return rc;
  2107. }catch(e){
  2108. if(!(e instanceof sqlite3.WasmAllocError)){
  2109. wasm.dealloc(wasm.peekPtr, pzErr);
  2110. wasm.pokePtr(pzErr, wasm.allocCString(e.message));
  2111. }
  2112. return VT.xError('xConnect',e);
  2113. }
  2114. },
  2115. xCreate: true /* just for testing. Will be removed afterwards. */,
  2116. xDisconnect: function(pVtab){
  2117. try {
  2118. VT.xVtab.unget(pVtab).dispose();
  2119. return 0;
  2120. }catch(e){
  2121. return VT.xError('xDisconnect',e);
  2122. }
  2123. },
  2124. xOpen: function(pVtab, ppCursor){
  2125. try{
  2126. const t = VT.xVtab.get(pVtab),
  2127. c = VT.xCursor.create(ppCursor);
  2128. T.assert(t instanceof capi.sqlite3_vtab)
  2129. .assert(c instanceof capi.sqlite3_vtab_cursor);
  2130. c._rowId = 0;
  2131. return 0;
  2132. }catch(e){
  2133. return VT.xError('xOpen',e);
  2134. }
  2135. },
  2136. xClose: function(pCursor){
  2137. try{
  2138. const c = VT.xCursor.unget(pCursor);
  2139. T.assert(c instanceof capi.sqlite3_vtab_cursor)
  2140. .assert(!VT.xCursor.get(pCursor));
  2141. c.dispose();
  2142. return 0;
  2143. }catch(e){
  2144. return VT.xError('xClose',e);
  2145. }
  2146. },
  2147. xNext: function(pCursor){
  2148. try{
  2149. const c = VT.xCursor.get(pCursor);
  2150. ++c._rowId;
  2151. return 0;
  2152. }catch(e){
  2153. return VT.xError('xNext',e);
  2154. }
  2155. },
  2156. xColumn: function(pCursor, pCtx, iCol){
  2157. try{
  2158. const c = VT.xCursor.get(pCursor);
  2159. switch(iCol){
  2160. case tmplCols.A:
  2161. capi.sqlite3_result_int(pCtx, 1000 + c._rowId);
  2162. break;
  2163. case tmplCols.B:
  2164. capi.sqlite3_result_int(pCtx, 2000 + c._rowId);
  2165. break;
  2166. default: sqlite3.SQLite3Error.toss("Invalid column id",iCol);
  2167. }
  2168. return 0;
  2169. }catch(e){
  2170. return VT.xError('xColumn',e);
  2171. }
  2172. },
  2173. xRowid: function(pCursor, ppRowid64){
  2174. try{
  2175. const c = VT.xCursor.get(pCursor);
  2176. VT.xRowid(ppRowid64, c._rowId);
  2177. return 0;
  2178. }catch(e){
  2179. return VT.xError('xRowid',e);
  2180. }
  2181. },
  2182. xEof: function(pCursor){
  2183. const c = VT.xCursor.get(pCursor),
  2184. rc = c._rowId>=10;
  2185. return rc;
  2186. },
  2187. xFilter: function(pCursor, idxNum, idxCStr,
  2188. argc, argv/* [sqlite3_value* ...] */){
  2189. try{
  2190. const c = VT.xCursor.get(pCursor);
  2191. c._rowId = 0;
  2192. const list = capi.sqlite3_values_to_js(argc, argv);
  2193. T.assert(argc === list.length);
  2194. //log(argc,"xFilter value(s):",list);
  2195. return 0;
  2196. }catch(e){
  2197. return VT.xError('xFilter',e);
  2198. }
  2199. },
  2200. xBestIndex: function(pVtab, pIdxInfo){
  2201. try{
  2202. //const t = VT.xVtab.get(pVtab);
  2203. const sii = capi.sqlite3_index_info;
  2204. const pii = new sii(pIdxInfo);
  2205. pii.$estimatedRows = 10;
  2206. pii.$estimatedCost = 10.0;
  2207. //log("xBestIndex $nConstraint =",pii.$nConstraint);
  2208. if(pii.$nConstraint>0){
  2209. // Validate nthConstraint() and nthConstraintUsage()
  2210. const max = pii.$nConstraint;
  2211. for(let i=0; i < max; ++i ){
  2212. let v = pii.nthConstraint(i,true);
  2213. T.assert(wasm.isPtr(v));
  2214. v = pii.nthConstraint(i);
  2215. T.assert(v instanceof sii.sqlite3_index_constraint)
  2216. .assert(v.pointer >= pii.$aConstraint);
  2217. v.dispose();
  2218. v = pii.nthConstraintUsage(i,true);
  2219. T.assert(wasm.isPtr(v));
  2220. v = pii.nthConstraintUsage(i);
  2221. T.assert(v instanceof sii.sqlite3_index_constraint_usage)
  2222. .assert(v.pointer >= pii.$aConstraintUsage);
  2223. v.$argvIndex = i;//just to get some values into xFilter
  2224. v.dispose();
  2225. }
  2226. }
  2227. //log("xBestIndex $nOrderBy =",pii.$nOrderBy);
  2228. if(pii.$nOrderBy>0){
  2229. // Validate nthOrderBy()
  2230. const max = pii.$nOrderBy;
  2231. for(let i=0; i < max; ++i ){
  2232. let v = pii.nthOrderBy(i,true);
  2233. T.assert(wasm.isPtr(v));
  2234. v = pii.nthOrderBy(i);
  2235. T.assert(v instanceof sii.sqlite3_index_orderby)
  2236. .assert(v.pointer >= pii.$aOrderBy);
  2237. v.dispose();
  2238. }
  2239. }
  2240. pii.dispose();
  2241. return 0;
  2242. }catch(e){
  2243. return VT.xError('xBestIndex',e);
  2244. }
  2245. }
  2246. }
  2247. });
  2248. this.db.onclose.disposeAfter.push(tmplMod);
  2249. T.assert(0===tmplMod.$xUpdate)
  2250. .assert(tmplMod.$xCreate)
  2251. .assert(tmplMod.$xCreate === tmplMod.$xConnect,
  2252. "setup() must make these equivalent and "+
  2253. "installMethods() must avoid re-compiling identical functions");
  2254. tmplMod.$xCreate = 0 /* make tmplMod eponymous-only */;
  2255. let rc = capi.sqlite3_create_module(
  2256. this.db, "testvtab", tmplMod, 0
  2257. );
  2258. this.db.checkRc(rc);
  2259. const list = this.db.selectArrays(
  2260. "SELECT a,b FROM testvtab where a<9999 and b>1 order by a, b"
  2261. /* Query is shaped so that it will ensure that some constraints
  2262. end up in xBestIndex(). */
  2263. );
  2264. T.assert(10===list.length)
  2265. .assert(1000===list[0][0])
  2266. .assert(2009===list[list.length-1][1]);
  2267. }
  2268. })/*custom vtab #1*/
  2269. ////////////////////////////////////////////////////////////////////////
  2270. .t({
  2271. name: 'virtual table #2: non-eponymous w/ automated exception wrapping',
  2272. predicate: (sqlite3)=>!!sqlite3.capi.sqlite3_vtab || "Missing vtab support",
  2273. test: function(sqlite3){
  2274. const VT = sqlite3.vtab;
  2275. const tmplCols = Object.assign(Object.create(null),{
  2276. A: 0, B: 1
  2277. });
  2278. /**
  2279. The vtab demonstrated here is a JS-ification of
  2280. ext/misc/templatevtab.c.
  2281. */
  2282. let throwOnCreate = 1 ? 0 : capi.SQLITE_CANTOPEN
  2283. /* ^^^ just for testing exception wrapping. Note that sqlite
  2284. always translates errors from a vtable to a generic
  2285. SQLITE_ERROR unless it's from xConnect()/xCreate() and that
  2286. callback sets an error string. */;
  2287. const vtabTrace = 1
  2288. ? ()=>{}
  2289. : (methodName,...args)=>console.debug('sqlite3_module::'+methodName+'():',...args);
  2290. const modConfig = {
  2291. /* catchExceptions changes how the methods are wrapped */
  2292. catchExceptions: true,
  2293. name: "vtab2test",
  2294. methods:{
  2295. xCreate: function(pDb, pAux, argc, argv, ppVtab, pzErr){
  2296. vtabTrace("xCreate",...arguments);
  2297. if(throwOnCreate){
  2298. sqlite3.SQLite3Error.toss(
  2299. throwOnCreate,
  2300. "Throwing a test exception."
  2301. );
  2302. }
  2303. const args = wasm.cArgvToJs(argc, argv);
  2304. vtabTrace("xCreate","argv:",args);
  2305. T.assert(args.length>=3);
  2306. const rc = capi.sqlite3_declare_vtab(
  2307. pDb, "CREATE TABLE ignored(a,b)"
  2308. );
  2309. if(0===rc){
  2310. const t = VT.xVtab.create(ppVtab);
  2311. T.assert(t === VT.xVtab.get(wasm.peekPtr(ppVtab)));
  2312. vtabTrace("xCreate",...arguments," ppVtab =",t.pointer);
  2313. }
  2314. return rc;
  2315. },
  2316. xConnect: true,
  2317. xDestroy: function(pVtab){
  2318. vtabTrace("xDestroy/xDisconnect",pVtab);
  2319. VT.xVtab.dispose(pVtab);
  2320. },
  2321. xDisconnect: true,
  2322. xOpen: function(pVtab, ppCursor){
  2323. const t = VT.xVtab.get(pVtab),
  2324. c = VT.xCursor.create(ppCursor);
  2325. T.assert(t instanceof capi.sqlite3_vtab)
  2326. .assert(c instanceof capi.sqlite3_vtab_cursor);
  2327. vtabTrace("xOpen",...arguments," cursor =",c.pointer);
  2328. c._rowId = 0;
  2329. },
  2330. xClose: function(pCursor){
  2331. vtabTrace("xClose",...arguments);
  2332. const c = VT.xCursor.unget(pCursor);
  2333. T.assert(c instanceof capi.sqlite3_vtab_cursor)
  2334. .assert(!VT.xCursor.get(pCursor));
  2335. c.dispose();
  2336. },
  2337. xNext: function(pCursor){
  2338. vtabTrace("xNext",...arguments);
  2339. const c = VT.xCursor.get(pCursor);
  2340. ++c._rowId;
  2341. },
  2342. xColumn: function(pCursor, pCtx, iCol){
  2343. vtabTrace("xColumn",...arguments);
  2344. const c = VT.xCursor.get(pCursor);
  2345. switch(iCol){
  2346. case tmplCols.A:
  2347. capi.sqlite3_result_int(pCtx, 1000 + c._rowId);
  2348. break;
  2349. case tmplCols.B:
  2350. capi.sqlite3_result_int(pCtx, 2000 + c._rowId);
  2351. break;
  2352. default: sqlite3.SQLite3Error.toss("Invalid column id",iCol);
  2353. }
  2354. },
  2355. xRowid: function(pCursor, ppRowid64){
  2356. vtabTrace("xRowid",...arguments);
  2357. const c = VT.xCursor.get(pCursor);
  2358. VT.xRowid(ppRowid64, c._rowId);
  2359. },
  2360. xEof: function(pCursor){
  2361. vtabTrace("xEof",...arguments);
  2362. return VT.xCursor.get(pCursor)._rowId>=10;
  2363. },
  2364. xFilter: function(pCursor, idxNum, idxCStr,
  2365. argc, argv/* [sqlite3_value* ...] */){
  2366. vtabTrace("xFilter",...arguments);
  2367. const c = VT.xCursor.get(pCursor);
  2368. c._rowId = 0;
  2369. const list = capi.sqlite3_values_to_js(argc, argv);
  2370. T.assert(argc === list.length);
  2371. },
  2372. xBestIndex: function(pVtab, pIdxInfo){
  2373. vtabTrace("xBestIndex",...arguments);
  2374. //const t = VT.xVtab.get(pVtab);
  2375. const pii = VT.xIndexInfo(pIdxInfo);
  2376. pii.$estimatedRows = 10;
  2377. pii.$estimatedCost = 10.0;
  2378. pii.dispose();
  2379. }
  2380. }/*methods*/
  2381. };
  2382. const tmplMod = VT.setupModule(modConfig);
  2383. T.assert(1===tmplMod.$iVersion);
  2384. this.db.onclose.disposeAfter.push(tmplMod);
  2385. this.db.checkRc(capi.sqlite3_create_module(
  2386. this.db.pointer, modConfig.name, tmplMod.pointer, 0
  2387. ));
  2388. this.db.exec([
  2389. "create virtual table testvtab2 using ",
  2390. modConfig.name,
  2391. "(arg1 blah, arg2 bloop)"
  2392. ]);
  2393. if(0){
  2394. /* If we DROP TABLE then xDestroy() is called. If the
  2395. vtab is instead destroyed when the db is closed,
  2396. xDisconnect() is called. */
  2397. this.db.onclose.disposeBefore.push(function(db){
  2398. console.debug("Explicitly dropping testvtab2 via disposeBefore handler...");
  2399. db.exec(
  2400. /** DROP TABLE is the only way to get xDestroy() to be called. */
  2401. "DROP TABLE testvtab2"
  2402. );
  2403. });
  2404. }
  2405. let list = this.db.selectArrays(
  2406. "SELECT a,b FROM testvtab2 where a<9999 and b>1 order by a, b"
  2407. /* Query is shaped so that it will ensure that some
  2408. constraints end up in xBestIndex(). */
  2409. );
  2410. T.assert(10===list.length)
  2411. .assert(1000===list[0][0])
  2412. .assert(2009===list[list.length-1][1]);
  2413. list = this.db.selectArrays(
  2414. "SELECT a,b FROM testvtab2 where a<9999 and b>1 order by b, a limit 5"
  2415. );
  2416. T.assert(5===list.length)
  2417. .assert(1000===list[0][0])
  2418. .assert(2004===list[list.length-1][1]);
  2419. // Call it as a table-valued function...
  2420. list = this.db.selectArrays([
  2421. "SELECT a,b FROM ", modConfig.name,
  2422. " where a<9999 and b>1 order by b, a limit 1"
  2423. ]);
  2424. T.assert(1===list.length)
  2425. .assert(1000===list[0][0])
  2426. .assert(2000===list[0][1]);
  2427. }
  2428. })/*custom vtab #2*/
  2429. ////////////////////////////////////////////////////////////////////////
  2430. .t('Custom collation', function(sqlite3){
  2431. let collationCounter = 0;
  2432. let myCmp = function(pArg,n1,p1,n2,p2){
  2433. //int (*)(void*,int,const void*,int,const void*)
  2434. ++collationCounter;
  2435. const rc = wasm.exports.sqlite3_strnicmp(p1,p2,(n1<n2?n1:n2));
  2436. return rc ? rc : (n1 - n2);
  2437. };
  2438. let rc = capi.sqlite3_create_collation_v2(this.db, "mycollation", capi.SQLITE_UTF8,
  2439. 0, myCmp, 0);
  2440. this.db.checkRc(rc);
  2441. rc = this.db.selectValue("select 'hi' = 'HI' collate mycollation");
  2442. T.assert(1===rc).assert(1===collationCounter);
  2443. rc = this.db.selectValue("select 'hii' = 'HI' collate mycollation");
  2444. T.assert(0===rc).assert(2===collationCounter);
  2445. rc = this.db.selectValue("select 'hi' = 'HIi' collate mycollation");
  2446. T.assert(0===rc).assert(3===collationCounter);
  2447. rc = capi.sqlite3_create_collation(this.db,"hi",capi.SQLITE_UTF8/*not enough args*/);
  2448. T.assert(capi.SQLITE_MISUSE === rc);
  2449. rc = capi.sqlite3_create_collation_v2(this.db,"hi",capi.SQLITE_UTF8+1/*invalid encoding*/,0,0,0);
  2450. T.assert(capi.SQLITE_FORMAT === rc)
  2451. .mustThrowMatching(()=>this.db.checkRc(rc),
  2452. /SQLITE_UTF8 is the only supported encoding./);
  2453. /*
  2454. We need to ensure that replacing that collation function does
  2455. the right thing. We don't have a handle to the underlying WASM
  2456. pointer from here, so cannot verify (without digging through
  2457. internal state) that the old one gets uninstalled, but we can
  2458. verify that a new one properly replaces it. (That said,
  2459. console.warn() output has shown that the uninstallation does
  2460. happen.)
  2461. */
  2462. collationCounter = 0;
  2463. myCmp = function(pArg,n1,p1,n2,p2){
  2464. --collationCounter;
  2465. return 0;
  2466. };
  2467. rc = capi.sqlite3_create_collation_v2(this.db, "MYCOLLATION", capi.SQLITE_UTF8,
  2468. 0, myCmp, 0);
  2469. this.db.checkRc(rc);
  2470. rc = this.db.selectValue("select 'hi' = 'HI' collate mycollation");
  2471. T.assert(rc>0).assert(-1===collationCounter);
  2472. rc = this.db.selectValue("select 'a' = 'b' collate mycollation");
  2473. T.assert(rc>0).assert(-2===collationCounter);
  2474. rc = capi.sqlite3_create_collation_v2(this.db, "MYCOLLATION", capi.SQLITE_UTF8,
  2475. 0, null, 0);
  2476. this.db.checkRc(rc);
  2477. rc = 0;
  2478. try {
  2479. this.db.selectValue("select 'a' = 'b' collate mycollation");
  2480. }catch(e){
  2481. /* Why is e.resultCode not automatically an extended result
  2482. code? The DB() class enables those automatically. */
  2483. rc = sqlite3.capi.sqlite3_extended_errcode(this.db);
  2484. }
  2485. T.assert(capi.SQLITE_ERROR_MISSING_COLLSEQ === rc);
  2486. })/*custom collation*/
  2487. ////////////////////////////////////////////////////////////////////////
  2488. .t('Close db', function(){
  2489. T.assert(this.db).assert(wasm.isPtr(this.db.pointer));
  2490. //wasm.sqlite3__wasm_db_reset(this.db); // will leak virtual tables!
  2491. this.db.close();
  2492. T.assert(!this.db.pointer);
  2493. })
  2494. ;/* end of oo1 checks */
  2495. ////////////////////////////////////////////////////////////////////////
  2496. T.g('kvvfs')
  2497. .t({
  2498. name: 'kvvfs is disabled in worker',
  2499. predicate: ()=>(isWorker() || "test is only valid in a Worker"),
  2500. test: function(sqlite3){
  2501. T.assert(
  2502. !capi.sqlite3_vfs_find('kvvfs'),
  2503. "Expecting kvvfs to be unregistered."
  2504. );
  2505. }
  2506. })
  2507. .t({
  2508. name: 'kvvfs in main thread',
  2509. predicate: ()=>(isUIThread()
  2510. || "local/sessionStorage are unavailable in a Worker"),
  2511. test: function(sqlite3){
  2512. const filename = this.kvvfsDbFile = 'session';
  2513. const pVfs = capi.sqlite3_vfs_find('kvvfs');
  2514. T.assert(pVfs);
  2515. const JDb = this.JDb = sqlite3.oo1.JsStorageDb;
  2516. const unlink = this.kvvfsUnlink = ()=>JDb.clearStorage(this.kvvfsDbFile);
  2517. unlink();
  2518. let db = new JDb(filename);
  2519. try {
  2520. db.exec([
  2521. 'create table kvvfs(a);',
  2522. 'insert into kvvfs(a) values(1),(2),(3)'
  2523. ]);
  2524. T.assert(3 === db.selectValue('select count(*) from kvvfs'));
  2525. db.close();
  2526. db = new JDb(filename);
  2527. db.exec('insert into kvvfs(a) values(4),(5),(6)');
  2528. T.assert(6 === db.selectValue('select count(*) from kvvfs'));
  2529. }finally{
  2530. db.close();
  2531. }
  2532. }
  2533. }/*kvvfs sanity checks*/)
  2534. //#if enable-see
  2535. .t({
  2536. name: 'kvvfs with SEE encryption',
  2537. predicate: ()=>(isUIThread()
  2538. || "Only available in main thread."),
  2539. test: function(sqlite3){
  2540. this.kvvfsUnlink();
  2541. let db;
  2542. const encOpt1 = 1
  2543. ? {textkey: 'foo'}
  2544. : {key: 'foo'};
  2545. const encOpt2 = encOpt1.textkey
  2546. ? encOpt1
  2547. : {hexkey: new Uint8Array([0x66,0x6f,0x6f]/*==>"foo"*/)}
  2548. try{
  2549. db = new this.JDb({
  2550. filename: this.kvvfsDbFile,
  2551. ...encOpt1
  2552. });
  2553. db.exec([
  2554. "create table t(a,b);",
  2555. "insert into t(a,b) values(1,2),(3,4)"
  2556. ]);
  2557. db.close();
  2558. let err;
  2559. try{
  2560. db = new this.JDb({
  2561. filename: this.kvvfsDbFile,
  2562. flags: 'ct'
  2563. });
  2564. T.assert(db) /* opening is fine, but... */;
  2565. db.exec("select 1 from sqlite_schema");
  2566. console.warn("sessionStorage =",sessionStorage);
  2567. }catch(e){
  2568. err = e;
  2569. }finally{
  2570. db.close();
  2571. }
  2572. T.assert(err,"Expecting an exception")
  2573. .assert(sqlite3.capi.SQLITE_NOTADB==err.resultCode,
  2574. "Expecting NOTADB");
  2575. db = new sqlite3.oo1.DB({
  2576. filename: this.kvvfsDbFile,
  2577. vfs: 'kvvfs',
  2578. ...encOpt2
  2579. });
  2580. T.assert( 4===db.selectValue('select sum(a) from t') );
  2581. }finally{
  2582. if( db ) db.close();
  2583. this.kvvfsUnlink();
  2584. }
  2585. }
  2586. })/*kvvfs with SEE*/
  2587. //#endif enable-see
  2588. ;/* end kvvfs tests */
  2589. ////////////////////////////////////////////////////////////////////////
  2590. T.g('Hook APIs')
  2591. .t({
  2592. name: "sqlite3_commit/rollback/update_hook()",
  2593. predicate: ()=>wasm.bigIntEnabled || "Update hook requires int64",
  2594. test: function(sqlite3){
  2595. let countCommit = 0, countRollback = 0;;
  2596. const db = new sqlite3.oo1.DB(':memory:',1 ? 'c' : 'ct');
  2597. let rc = capi.sqlite3_commit_hook(db, (p)=>{
  2598. ++countCommit;
  2599. return (1 === p) ? 0 : capi.SQLITE_ERROR;
  2600. }, 1);
  2601. T.assert( 0 === rc /*void pointer*/ );
  2602. // Commit hook...
  2603. T.assert( 0!=capi.sqlite3_get_autocommit(db) );
  2604. db.exec("BEGIN; SELECT 1; COMMIT");
  2605. T.assert(0 === countCommit,
  2606. "No-op transactions (mostly) do not trigger commit hook.");
  2607. db.exec("BEGIN EXCLUSIVE; SELECT 1; COMMIT");
  2608. T.assert(1 === countCommit,
  2609. "But EXCLUSIVE transactions do.");
  2610. db.transaction((d)=>{
  2611. T.assert( 0==capi.sqlite3_get_autocommit(db) );
  2612. d.exec("create table t(a)");
  2613. });
  2614. T.assert(2 === countCommit);
  2615. // Rollback hook:
  2616. rc = capi.sqlite3_rollback_hook(db, (p)=>{
  2617. ++countRollback;
  2618. T.assert( 2 === p );
  2619. }, 2);
  2620. T.assert( 0 === rc /*void pointer*/ );
  2621. T.mustThrowMatching(()=>{
  2622. db.transaction('drop table t',()=>{})
  2623. }, (e)=>{
  2624. return (capi.SQLITE_MISUSE === e.resultCode)
  2625. && ( e.message.indexOf('Invalid argument') > 0 );
  2626. });
  2627. T.assert(0 === countRollback, "Transaction was not started.");
  2628. T.mustThrowMatching(()=>{
  2629. db.transaction('immediate', ()=>{
  2630. sqlite3.SQLite3Error.toss(capi.SQLITE_FULL,'testing rollback hook');
  2631. });
  2632. }, (e)=>{
  2633. return capi.SQLITE_FULL === e.resultCode
  2634. });
  2635. T.assert(1 === countRollback);
  2636. // Update hook...
  2637. const countUpdate = Object.create(null);
  2638. capi.sqlite3_update_hook(db, (p,op,dbName,tbl,rowid)=>{
  2639. T.assert('main' === dbName.toLowerCase())
  2640. .assert('t' === tbl.toLowerCase())
  2641. .assert(3===p)
  2642. .assert('bigint' === typeof rowid);
  2643. switch(op){
  2644. case capi.SQLITE_INSERT:
  2645. case capi.SQLITE_UPDATE:
  2646. case capi.SQLITE_DELETE:
  2647. countUpdate[op] = (countUpdate[op]||0) + 1;
  2648. break;
  2649. default: toss("Unexpected hook operator:",op);
  2650. }
  2651. }, 3);
  2652. db.transaction((d)=>{
  2653. d.exec([
  2654. "insert into t(a) values(1);",
  2655. "update t set a=2;",
  2656. "update t set a=3;",
  2657. "delete from t where a=3"
  2658. // update hook is not called for an unqualified DELETE
  2659. ]);
  2660. });
  2661. T.assert(1 === countRollback)
  2662. .assert(3 === countCommit)
  2663. .assert(1 === countUpdate[capi.SQLITE_INSERT])
  2664. .assert(2 === countUpdate[capi.SQLITE_UPDATE])
  2665. .assert(1 === countUpdate[capi.SQLITE_DELETE]);
  2666. //wasm.xWrap.FuncPtrAdapter.debugFuncInstall = true;
  2667. T.assert(1 === capi.sqlite3_commit_hook(db, 0, 0));
  2668. T.assert(2 === capi.sqlite3_rollback_hook(db, 0, 0));
  2669. T.assert(3 === capi.sqlite3_update_hook(db, 0, 0));
  2670. //wasm.xWrap.FuncPtrAdapter.debugFuncInstall = false;
  2671. db.close();
  2672. }
  2673. })/* commit/rollback/update hooks */
  2674. .t({
  2675. name: "sqlite3_preupdate_hook()",
  2676. predicate: ()=>capi.sqlite3_preupdate_hook || "Missing pre-update hook API",
  2677. test: function(sqlite3){
  2678. const db = new sqlite3.oo1.DB(':memory:', 1 ? 'c' : 'ct');
  2679. const countHook = Object.create(null);
  2680. let rc = capi.sqlite3_preupdate_hook(
  2681. db, function(p, pDb, op, zDb, zTbl, iKey1, iKey2){
  2682. T.assert(9 === p)
  2683. .assert(db.pointer === pDb)
  2684. .assert(1 === capi.sqlite3_preupdate_count(pDb))
  2685. .assert( 0 > capi.sqlite3_preupdate_blobwrite(pDb) );
  2686. countHook[op] = (countHook[op]||0) + 1;
  2687. switch(op){
  2688. case capi.SQLITE_INSERT:
  2689. case capi.SQLITE_UPDATE:
  2690. T.assert('number' === typeof capi.sqlite3_preupdate_new_js(pDb, 0));
  2691. break;
  2692. case capi.SQLITE_DELETE:
  2693. T.assert('number' === typeof capi.sqlite3_preupdate_old_js(pDb, 0));
  2694. break;
  2695. default: toss("Unexpected hook operator:",op);
  2696. }
  2697. },
  2698. 9
  2699. );
  2700. db.transaction((d)=>{
  2701. d.exec([
  2702. "create table t(a);",
  2703. "insert into t(a) values(1);",
  2704. "update t set a=2;",
  2705. "update t set a=3;",
  2706. "delete from t where a=3"
  2707. ]);
  2708. });
  2709. T.assert(1 === countHook[capi.SQLITE_INSERT])
  2710. .assert(2 === countHook[capi.SQLITE_UPDATE])
  2711. .assert(1 === countHook[capi.SQLITE_DELETE]);
  2712. //wasm.xWrap.FuncPtrAdapter.debugFuncInstall = true;
  2713. db.close();
  2714. //wasm.xWrap.FuncPtrAdapter.debugFuncInstall = false;
  2715. }
  2716. })/*pre-update hooks*/
  2717. ;/*end hook API tests*/
  2718. ////////////////////////////////////////////////////////////////////////
  2719. T.g('Auto-extension API')
  2720. .t({
  2721. name: "Auto-extension sanity checks.",
  2722. test: function(sqlite3){
  2723. let counter = 0;
  2724. const fp = wasm.installFunction('i(ppp)', function(pDb,pzErr,pApi){
  2725. ++counter;
  2726. return 0;
  2727. });
  2728. (new sqlite3.oo1.DB()).close();
  2729. T.assert( 0===counter );
  2730. capi.sqlite3_auto_extension(fp);
  2731. (new sqlite3.oo1.DB()).close();
  2732. T.assert( 1===counter );
  2733. (new sqlite3.oo1.DB()).close();
  2734. T.assert( 2===counter );
  2735. capi.sqlite3_cancel_auto_extension(fp);
  2736. wasm.uninstallFunction(fp);
  2737. (new sqlite3.oo1.DB()).close();
  2738. T.assert( 2===counter );
  2739. }
  2740. });
  2741. ////////////////////////////////////////////////////////////////////////
  2742. T.g('Session API')
  2743. .t({
  2744. name: 'Session API sanity checks',
  2745. predicate: ()=>!!capi.sqlite3changegroup_add || "Missing session API",
  2746. test: function(sqlite3){
  2747. //warn("The session API tests could use some expansion.");
  2748. const db1 = new sqlite3.oo1.DB(), db2 = new sqlite3.oo1.DB();
  2749. const sqlInit = [
  2750. "create table t(rowid INTEGER PRIMARY KEY,a,b); ",
  2751. "insert into t(rowid,a,b) values",
  2752. "(1,'a1','b1'),",
  2753. "(2,'a2','b2'),",
  2754. "(3,'a3','b3');"
  2755. ].join('');
  2756. db1.exec(sqlInit);
  2757. db2.exec(sqlInit);
  2758. T.assert(3 === db1.selectValue("select count(*) from t"))
  2759. .assert('b3' === db1.selectValue('select b from t where rowid=3'));
  2760. const stackPtr = wasm.pstack.pointer;
  2761. try{
  2762. let ppOut = wasm.pstack.allocPtr();
  2763. let rc = capi.sqlite3session_create(db1, "main", ppOut);
  2764. T.assert(0===rc);
  2765. let pSession = wasm.peekPtr(ppOut);
  2766. T.assert(pSession && wasm.isPtr(pSession));
  2767. capi.sqlite3session_table_filter(pSession, (pCtx, tbl)=>{
  2768. T.assert('t' === tbl).assert( 99 === pCtx );
  2769. return 1;
  2770. }, 99);
  2771. db1.exec([
  2772. "update t set b='bTwo' where rowid=2;",
  2773. "update t set a='aThree' where rowid=3;",
  2774. "delete from t where rowid=1;",
  2775. "insert into t(rowid,a,b) values(4,'a4','b4')"
  2776. ]);
  2777. T.assert('bTwo' === db1.selectValue("select b from t where rowid=2"))
  2778. .assert(undefined === db1.selectValue('select a from t where rowid=1'))
  2779. .assert('b4' === db1.selectValue('select b from t where rowid=4'))
  2780. .assert(3 === db1.selectValue('select count(*) from t'));
  2781. const testSessionEnable =
  2782. false /* it's not yet clear whether these test failures are
  2783. broken tests or broken bindings. */;
  2784. if(testSessionEnable){
  2785. rc = capi.sqlite3session_enable(pSession, 0);
  2786. T.assert( 0 === rc )
  2787. .assert( 0 === capi.sqlite3session_enable(pSession, -1) );
  2788. db1.exec("delete from t where rowid=2;");
  2789. rc = capi.sqlite3session_enable(pSession, 1);
  2790. T.assert( rc > 0 )
  2791. .assert( capi.sqlite3session_enable(pSession, -1) > 0 )
  2792. .assert(undefined === db1.selectValue('select a from t where rowid=2'));
  2793. }else{
  2794. //warn("sqlite3session_enable() tests are currently disabled.");
  2795. }
  2796. let db1Count = db1.selectValue("select count(*) from t");
  2797. T.assert( db1Count === (testSessionEnable ? 2 : 3) );
  2798. /* Capture changeset and destroy session. */
  2799. let pnChanges = wasm.pstack.alloc('i32'),
  2800. ppChanges = wasm.pstack.allocPtr();
  2801. rc = capi.sqlite3session_changeset(pSession, pnChanges, ppChanges);
  2802. T.assert( 0 === rc );
  2803. capi.sqlite3session_delete(pSession);
  2804. pSession = 0;
  2805. const pChanges = wasm.peekPtr(ppChanges),
  2806. nChanges = wasm.peek32(pnChanges);
  2807. T.assert( pChanges && wasm.isPtr( pChanges ) )
  2808. .assert( nChanges > 0 );
  2809. /* Revert db1 via an inverted changeset, but keep pChanges
  2810. and nChanges for application to db2. */
  2811. rc = capi.sqlite3changeset_invert( nChanges, pChanges, pnChanges, ppChanges );
  2812. T.assert( 0 === rc );
  2813. rc = capi.sqlite3changeset_apply(
  2814. db1, wasm.peek32(pnChanges), wasm.peekPtr(ppChanges), 0, (pCtx, eConflict, pIter)=>{
  2815. return 1;
  2816. }, 0
  2817. );
  2818. T.assert( 0 === rc );
  2819. wasm.dealloc( wasm.peekPtr(ppChanges) );
  2820. pnChanges = ppChanges = 0;
  2821. T.assert('b2' === db1.selectValue("select b from t where rowid=2"))
  2822. .assert('a1' === db1.selectValue('select a from t where rowid=1'))
  2823. .assert(undefined === db1.selectValue('select b from t where rowid=4'));
  2824. db1Count = db1.selectValue("select count(*) from t");
  2825. T.assert(3 === db1Count);
  2826. /* Apply pre-reverted changeset (pChanges, nChanges) to
  2827. db2... */
  2828. rc = capi.sqlite3changeset_apply(
  2829. db2, nChanges, pChanges, 0, (pCtx, eConflict, pIter)=>{
  2830. return pCtx ? 1 : 0
  2831. }, 1
  2832. );
  2833. wasm.dealloc( pChanges );
  2834. T.assert( 0 === rc )
  2835. .assert( 'b4' === db2.selectValue('select b from t where rowid=4') )
  2836. .assert( 'aThree' === db2.selectValue('select a from t where rowid=3') )
  2837. .assert( undefined === db2.selectValue('select b from t where rowid=1') );
  2838. if(testSessionEnable){
  2839. T.assert( (undefined === db2.selectValue('select b from t where rowid=2')),
  2840. "But... the session was disabled when rowid=2 was deleted?" );
  2841. log("rowids from db2.t:",db2.selectValues('select rowid from t order by rowid'));
  2842. T.assert( 3 === db2.selectValue('select count(*) from t') );
  2843. }else{
  2844. T.assert( 'bTwo' === db2.selectValue('select b from t where rowid=2') )
  2845. .assert( 3 === db2.selectValue('select count(*) from t') );
  2846. }
  2847. }finally{
  2848. wasm.pstack.restore(stackPtr);
  2849. db1.close();
  2850. db2.close();
  2851. }
  2852. }
  2853. })/*session API sanity tests*/
  2854. ;/*end of session API group*/;
  2855. ////////////////////////////////////////////////////////////////////////
  2856. T.g('OPFS: Origin-Private File System',
  2857. (sqlite3)=>(sqlite3.capi.sqlite3_vfs_find("opfs")
  2858. || 'requires "opfs" VFS'))
  2859. .t({
  2860. name: 'OPFS db sanity checks',
  2861. test: async function(sqlite3){
  2862. T.assert(capi.sqlite3_vfs_find('opfs'));
  2863. const opfs = sqlite3.opfs;
  2864. const filename = this.opfsDbFile = '/dir/sqlite3-tester1.db';
  2865. const fileUri = 'file://'+filename+'?delete-before-open=1';
  2866. const initSql = [
  2867. 'create table p(a);',
  2868. 'insert into p(a) values(1),(2),(3)'
  2869. ];
  2870. let db = new sqlite3.oo1.OpfsDb(fileUri);
  2871. try {
  2872. db.exec(initSql);
  2873. T.assert(3 === db.selectValue('select count(*) from p'));
  2874. db.close();
  2875. db = new sqlite3.oo1.OpfsDb(filename);
  2876. db.exec('insert into p(a) values(4),(5),(6)');
  2877. T.assert(6 === db.selectValue('select count(*) from p'));
  2878. this.opfsDbExport = capi.sqlite3_js_db_export(db);
  2879. T.assert(this.opfsDbExport instanceof Uint8Array)
  2880. .assert(this.opfsDbExport.byteLength>0
  2881. && 0===this.opfsDbExport.byteLength % 512);
  2882. }finally{
  2883. db.close();
  2884. }
  2885. T.assert(await opfs.entryExists(filename));
  2886. try {
  2887. db = new sqlite3.oo1.OpfsDb(fileUri);
  2888. db.exec(initSql) /* will throw if delete-before-open did not work */;
  2889. T.assert(3 === db.selectValue('select count(*) from p'));
  2890. }finally{
  2891. if(db) db.close();
  2892. }
  2893. }
  2894. }/*OPFS db sanity checks*/)
  2895. .t({
  2896. name: 'OPFS import',
  2897. test: async function(sqlite3){
  2898. let db;
  2899. const filename = this.opfsDbFile;
  2900. try {
  2901. const exp = this.opfsDbExport;
  2902. delete this.opfsDbExport;
  2903. this.opfsImportSize = await sqlite3.oo1.OpfsDb.importDb(filename, exp);
  2904. db = new sqlite3.oo1.OpfsDb(this.opfsDbFile);
  2905. T.assert(6 === db.selectValue('select count(*) from p')).
  2906. assert( this.opfsImportSize == exp.byteLength );
  2907. db.close();
  2908. const unlink = this.opfsUnlink =
  2909. (fn=filename)=>sqlite3.util.sqlite3__wasm_vfs_unlink("opfs",fn);
  2910. this.opfsUnlink(filename);
  2911. T.assert(!(await sqlite3.opfs.entryExists(filename)));
  2912. // Try again with a function as an input source:
  2913. let cursor = 0;
  2914. const blockSize = 512, end = exp.byteLength;
  2915. const reader = async function(){
  2916. if(cursor >= exp.byteLength){
  2917. return undefined;
  2918. }
  2919. const rv = exp.subarray(cursor, cursor+blockSize>end ? end : cursor+blockSize);
  2920. cursor += blockSize;
  2921. return rv;
  2922. };
  2923. this.opfsImportSize = await sqlite3.oo1.OpfsDb.importDb(filename, reader);
  2924. db = new sqlite3.oo1.OpfsDb(this.opfsDbFile);
  2925. T.assert(6 === db.selectValue('select count(*) from p')).
  2926. assert( this.opfsImportSize == exp.byteLength );
  2927. }finally{
  2928. if(db) db.close();
  2929. }
  2930. }
  2931. }/*OPFS export/import*/)
  2932. .t({
  2933. name: '(Internal-use) OPFS utility APIs',
  2934. test: async function(sqlite3){
  2935. const filename = this.opfsDbFile;
  2936. const unlink = this.opfsUnlink;
  2937. T.assert(filename && !!unlink);
  2938. delete this.opfsDbFile;
  2939. delete this.opfsUnlink;
  2940. /**************************************************************
  2941. ATTENTION CLIENT-SIDE USERS: sqlite3.opfs is NOT intended
  2942. for client-side use. It is only for this project's own
  2943. internal use. Its APIs are subject to change or removal at
  2944. any time.
  2945. ***************************************************************/
  2946. const opfs = sqlite3.opfs;
  2947. const fSize = this.opfsImportSize;
  2948. delete this.opfsImportSize;
  2949. let sh;
  2950. try{
  2951. T.assert(await opfs.entryExists(filename));
  2952. const [dirHandle, filenamePart] = await opfs.getDirForFilename(filename, false);
  2953. const fh = await dirHandle.getFileHandle(filenamePart);
  2954. sh = await fh.createSyncAccessHandle();
  2955. T.assert(fSize === await sh.getSize());
  2956. await sh.close();
  2957. sh = undefined;
  2958. unlink();
  2959. T.assert(!(await opfs.entryExists(filename)));
  2960. }finally{
  2961. if(sh) await sh.close();
  2962. unlink();
  2963. }
  2964. // Some sanity checks of the opfs utility functions...
  2965. const testDir = '/sqlite3-opfs-'+opfs.randomFilename(12);
  2966. const aDir = testDir+'/test/dir';
  2967. T.assert(await opfs.mkdir(aDir), "mkdir failed")
  2968. .assert(await opfs.mkdir(aDir), "mkdir must pass if the dir exists")
  2969. .assert(!(await opfs.unlink(testDir+'/test')), "delete 1 should have failed (dir not empty)")
  2970. .assert((await opfs.unlink(testDir+'/test/dir')), "delete 2 failed")
  2971. .assert(!(await opfs.unlink(testDir+'/test/dir')),
  2972. "delete 2b should have failed (dir already deleted)")
  2973. .assert((await opfs.unlink(testDir, true)), "delete 3 failed")
  2974. .assert(!(await opfs.entryExists(testDir)),
  2975. "entryExists(",testDir,") should have failed");
  2976. }
  2977. }/*OPFS util sanity checks*/)
  2978. ;/* end OPFS tests */
  2979. ////////////////////////////////////////////////////////////////////////
  2980. T.g('OPFS SyncAccessHandle Pool VFS',
  2981. (sqlite3)=>(hasOpfs() || "requires OPFS APIs"))
  2982. .t({
  2983. name: 'SAH sanity checks',
  2984. test: async function(sqlite3){
  2985. T.assert(!sqlite3.capi.sqlite3_vfs_find(sahPoolConfig.name))
  2986. .assert(sqlite3.capi.sqlite3_js_vfs_list().indexOf(sahPoolConfig.name) < 0)
  2987. const inst = sqlite3.installOpfsSAHPoolVfs,
  2988. catcher = (e)=>{
  2989. error("Cannot load SAH pool VFS.",
  2990. "This might not be a problem,",
  2991. "depending on the environment.");
  2992. return false;
  2993. };
  2994. let u1, u2;
  2995. // Ensure that two immediately-consecutive installations
  2996. // resolve to the same Promise instead of triggering
  2997. // a locking error.
  2998. const P1 = inst(sahPoolConfig).then(u=>u1 = u).catch(catcher),
  2999. P2 = inst(sahPoolConfig).then(u=>u2 = u).catch(catcher);
  3000. await Promise.all([P1, P2]);
  3001. if(!(await P1)) return;
  3002. T.assert(u1 === u2)
  3003. .assert(sahPoolConfig.name === u1.vfsName)
  3004. .assert(sqlite3.capi.sqlite3_vfs_find(sahPoolConfig.name))
  3005. .assert(u1.getCapacity() >= sahPoolConfig.initialCapacity
  3006. /* If a test fails before we get to nuke the VFS, we
  3007. can have more than the initial capacity on the next
  3008. run. */)
  3009. .assert(u1.getCapacity() + 2 === (await u2.addCapacity(2)))
  3010. .assert(2 === (await u2.reduceCapacity(2)))
  3011. .assert(sqlite3.capi.sqlite3_js_vfs_list().indexOf(sahPoolConfig.name) >= 0);
  3012. T.assert(0 === u1.getFileCount());
  3013. const dbName = '/foo.db';
  3014. let db = new u1.OpfsSAHPoolDb(dbName);
  3015. T.assert(db instanceof sqlite3.oo1.DB)
  3016. .assert(1 === u1.getFileCount());
  3017. db.exec([
  3018. 'pragma locking_mode=exclusive;',
  3019. 'pragma journal_mode=wal;'
  3020. /* WAL mode only works in this VFS if locking_mode=exclusive
  3021. is invoked prior to the first db access, as this build
  3022. does not have the shared-memory APIs needed for WAL without
  3023. exclusive-mode locking. See:
  3024. https://sqlite.org/wal.html#use_of_wal_without_shared_memory
  3025. Note that WAL mode here DOES NOT add any concurrency capabilities
  3026. to this VFS, but it MAY provide slightly improved performance
  3027. over the other journaling modes.
  3028. */,
  3029. 'create table t(a);',
  3030. 'insert into t(a) values(1),(2),(3)'
  3031. ]);
  3032. T.assert(2 === u1.getFileCount() /* one is the journal file */)
  3033. .assert(3 === db.selectValue('select count(*) from t'))
  3034. .assert(
  3035. 'wal'===db.selectValue('pragma journal_mode')
  3036. || wasm.compileOptionUsed('OMIT_WAL')
  3037. );
  3038. db.close();
  3039. T.assert(1 === u1.getFileCount());
  3040. db = new u2.OpfsSAHPoolDb(dbName);
  3041. T.assert(1 === u1.getFileCount());
  3042. db.close();
  3043. const fileNames = u1.getFileNames();
  3044. T.assert(1 === fileNames.length)
  3045. .assert(dbName === fileNames[0])
  3046. .assert(1 === u1.getFileCount())
  3047. if(1){ // test exportFile() and importDb()
  3048. const dbytes = u1.exportFile(dbName);
  3049. T.assert(dbytes.length >= 4096);
  3050. const dbName2 = '/exported.db';
  3051. let nWrote = u1.importDb(dbName2, dbytes);
  3052. T.assert( 2 == u1.getFileCount() )
  3053. .assert( dbytes.byteLength == nWrote );
  3054. let db2 = new u1.OpfsSAHPoolDb(dbName2);
  3055. T.assert(db2 instanceof sqlite3.oo1.DB)
  3056. .assert('wal' !== db2.selectValue("pragma journal_mode")
  3057. /* importDb() unsets the WAL-mode header for
  3058. historical reasons. Because clients must
  3059. explicitly enable pragma locking_mode=exclusive
  3060. before using WAL, that behavior is retained. */)
  3061. .assert(3 === db2.selectValue('select count(*) from t'));
  3062. db2.close();
  3063. T.assert(true === u1.unlink(dbName2))
  3064. .assert(false === u1.unlink(dbName2))
  3065. .assert(1 === u1.getFileCount())
  3066. .assert(1 === u1.getFileNames().length);
  3067. // Try again with a function as an input source:
  3068. let cursor = 0;
  3069. const blockSize = 1024, end = dbytes.byteLength;
  3070. const reader = async function(){
  3071. if(cursor >= dbytes.byteLength){
  3072. return undefined;
  3073. }
  3074. const rv = dbytes.subarray(cursor, cursor+blockSize>end ? end : cursor+blockSize);
  3075. cursor += blockSize;
  3076. return rv;
  3077. };
  3078. nWrote = await u1.importDb(dbName2, reader);
  3079. T.assert( 2 == u1.getFileCount() );
  3080. db2 = new u1.OpfsSAHPoolDb(dbName2);
  3081. T.assert(db2 instanceof sqlite3.oo1.DB)
  3082. .assert(3 === db2.selectValue('select count(*) from t'));
  3083. db2.close();
  3084. T.assert(true === u1.unlink(dbName2))
  3085. .assert(dbytes.byteLength == nWrote);
  3086. }
  3087. T.assert(true === u1.unlink(dbName))
  3088. .assert(false === u1.unlink(dbName))
  3089. .assert(0 === u1.getFileCount())
  3090. .assert(0 === u1.getFileNames().length);
  3091. // Demonstrate that two SAH pools can coexist so long as
  3092. // they have different names.
  3093. const conf2 = JSON.parse(JSON.stringify(sahPoolConfig));
  3094. conf2.name += '-test2';
  3095. const POther = await inst(conf2);
  3096. //log("Installed second SAH instance as",conf2.name);
  3097. T.assert(0 === POther.getFileCount())
  3098. .assert(true === await POther.removeVfs());
  3099. if(0){
  3100. /* Enable this block to inspect vfs's contents via the dev
  3101. console or OPFS Explorer browser extension. The
  3102. following bits will remove them. */
  3103. return;
  3104. }
  3105. T.assert(true === await u2.removeVfs())
  3106. .assert(false === await u1.removeVfs())
  3107. .assert(!sqlite3.capi.sqlite3_vfs_find(sahPoolConfig.name));
  3108. let cErr, u3;
  3109. conf2.$testThrowPhase2 = new Error("Testing throwing during init.");
  3110. conf2.name = sahPoolConfig.name+'-err';
  3111. const P3 = await inst(conf2).then(u=>u3 = u).catch((e)=>cErr=e);
  3112. T.assert(P3 === conf2.$testThrowPhase2)
  3113. .assert(cErr === P3)
  3114. .assert(undefined === u3)
  3115. .assert(!sqlite3.capi.sqlite3_vfs_find(conf2.name));
  3116. delete conf2.$testThrowPhase2;
  3117. T.assert(cErr === await inst(conf2).catch(e=>e),
  3118. "Init result is cached even if it failed");
  3119. /* Ensure that the forceReinitIfPreviouslyFailed fallback bypasses
  3120. the VFS init cache... */
  3121. cErr = u3 = undefined;
  3122. conf2.forceReinitIfPreviouslyFailed = true;
  3123. conf2.verbosity = 3;
  3124. const P3b = await inst(conf2).then(u=>u3 = u).catch((e)=>cErr=e);
  3125. T.assert(undefined === cErr)
  3126. .assert(P3b === u3)
  3127. .assert(P3b === await inst(conf2))
  3128. .assert(true === await u3.removeVfs())
  3129. .assert(false === await P3b.removeVfs());
  3130. }
  3131. }/*OPFS SAH Pool sanity checks*/)
  3132. ////////////////////////////////////////////////////////////////////////
  3133. T.g('Misc. APIs')
  3134. .t('bind_parameter_...', function(sqlite3){
  3135. const db = new sqlite3.oo1.DB();
  3136. db.exec("create table t(a)");
  3137. const stmt = db.prepare("insert into t(a) values($a)");
  3138. T.assert( 1===capi.sqlite3_bind_parameter_count(stmt) )
  3139. .assert( 1===capi.sqlite3_bind_parameter_index(stmt, "$a") )
  3140. .assert( 0===capi.sqlite3_bind_parameter_index(stmt, ":a") )
  3141. .assert( 1===stmt.getParamIndex("$a") )
  3142. .assert( 0===stmt.getParamIndex(":a") )
  3143. .assert( "$a"===capi.sqlite3_bind_parameter_name(stmt, 1) )
  3144. .assert( null===capi.sqlite3_bind_parameter_name(stmt, 0) )
  3145. .assert( "$a"===stmt.getParamName(1) )
  3146. .assert( null===stmt.getParamName(0) );
  3147. stmt.finalize();
  3148. db.close();
  3149. })
  3150. ////////////////////////////////////////////////////////////////////
  3151. .t("Misc. stmt_...", function(sqlite3){
  3152. const db = new sqlite3.oo1.DB();
  3153. db.exec("create table t(a doggiebiscuits); insert into t(a) values(123)");
  3154. const stmt = db.prepare("select a, a+1 from t");
  3155. T.assert( stmt.isReadOnly() )
  3156. .assert( 0===capi.sqlite3_stmt_isexplain(stmt) )
  3157. .assert( 0===capi.sqlite3_stmt_explain(stmt, 1) )
  3158. .assert( 0!==capi.sqlite3_stmt_isexplain(stmt) )
  3159. .assert( 0===capi.sqlite3_stmt_explain(stmt, 2) )
  3160. .assert( 0!==capi.sqlite3_stmt_isexplain(stmt) )
  3161. .assert( 0===capi.sqlite3_stmt_explain(stmt, 0) )
  3162. .assert( 0===capi.sqlite3_stmt_isexplain(stmt) );
  3163. let n = 0;
  3164. while( capi.SQLITE_ROW === capi.sqlite3_step(stmt) ){
  3165. ++n;
  3166. T.assert( 0!==capi.sqlite3_stmt_explain(stmt, 1),
  3167. "Because stmt is busy" )
  3168. .assert( capi.sqlite3_stmt_busy(stmt) )
  3169. .assert( stmt.isBusy() )
  3170. .assert( 0!==capi.sqlite3_stmt_readonly(stmt) )
  3171. .assert( true===stmt.isReadOnly() );
  3172. const sv = capi.sqlite3_column_value(stmt, 0);
  3173. T.assert( 123===capi.sqlite3_value_int(sv) )
  3174. .assert( "doggiebiscuits"===capi.sqlite3_column_decltype(stmt,0) )
  3175. .assert( null===capi.sqlite3_column_decltype(stmt,1) );
  3176. }
  3177. T.assert( 1===n )
  3178. .assert( 0===capi.sqlite3_stmt_busy(stmt) )
  3179. .assert( !stmt.isBusy() );
  3180. stmt.finalize();
  3181. db.close();
  3182. })
  3183. ////////////////////////////////////////////////////////////////////
  3184. .t("interrupt", function(sqlite3){
  3185. const db = new sqlite3.oo1.DB();
  3186. T.assert( 0===capi.sqlite3_is_interrupted(db) );
  3187. capi.sqlite3_interrupt(db);
  3188. T.assert( 0!==capi.sqlite3_is_interrupted(db) );
  3189. db.close();
  3190. })
  3191. ////////////////////////////////////////////////////////////////////////
  3192. T.g('Bug Reports')
  3193. .t({
  3194. name: 'Delete via bound parameter in subquery',
  3195. predicate: ()=>wasm.compileOptionUsed('ENABLE_FTS5') || "Missing FTS5",
  3196. test: function(sqlite3){
  3197. // Testing https://sqlite.org/forum/forumpost/40ce55bdf5
  3198. // with the exception that that post uses "external content"
  3199. // for the FTS index.
  3200. const db = new sqlite3.oo1.DB();//(':memory:','wt');
  3201. db.exec([
  3202. "create virtual table f using fts5 (path);",
  3203. "insert into f(path) values('abc'),('def'),('ghi');"
  3204. ]);
  3205. const fetchEm = ()=> db.exec({
  3206. sql: "SELECT * FROM f order by path",
  3207. rowMode: 'array'
  3208. });
  3209. const dump = function(lbl){
  3210. let rc = fetchEm();
  3211. log((lbl ? (lbl+' results') : ''),rc);
  3212. };
  3213. //dump('Full fts table');
  3214. let rc = fetchEm();
  3215. T.assert(3===rc.length);
  3216. db.exec(`
  3217. delete from f where rowid in (
  3218. select rowid from f where path = :path
  3219. )`,
  3220. {bind: {":path": "def"}}
  3221. );
  3222. //dump('After deleting one entry via subquery');
  3223. rc = fetchEm();
  3224. T.assert(2===rc.length)
  3225. .assert('abcghi'===rc.join(''));
  3226. //log('rc =',rc);
  3227. db.close();
  3228. }
  3229. })
  3230. .t({
  3231. name: 'r/o connection recovery from write op error',
  3232. predicate: ()=>hasOpfs() || "Requires OPFS to reproduce",
  3233. //predicate: ()=>false,
  3234. test: async function(sqlite3){
  3235. /* https://sqlite.org/forum/forumpost/cf37d5ff1182c31081
  3236. The "opfs" VFS (but not SAHPool) was formerly misbehaving
  3237. after a write attempt was made on a db opened with
  3238. mode=ro. This test ensures that that behavior is fixed and
  3239. compares that behavior with other VFSes. */
  3240. const tryOne = function(vfsName,descr){
  3241. const uri = 'file:///foo.db';
  3242. let db = new sqlite3.oo1.DB(uri + (vfsName ? '?vfs='+vfsName : ''));
  3243. db.exec([
  3244. "drop table if exists t;",
  3245. "create table t(a);",
  3246. "insert into t(a) values('abc'),('def'),('ghi');"
  3247. ]);
  3248. db.close();
  3249. db = new sqlite3.oo1.DB(uri+'?mode=ro'+
  3250. (vfsName ? '&vfs='+vfsName : ''));
  3251. let err;
  3252. try {
  3253. db.exec('insert into t(a) values(1)');
  3254. }catch(e){
  3255. err = e;
  3256. }
  3257. T.assert(err && (err.message.indexOf('SQLITE_READONLY')===0));
  3258. try{
  3259. db.exec('select a from t');
  3260. }finally{
  3261. db.close();
  3262. }
  3263. };
  3264. const poolConfig = JSON.parse(JSON.stringify(sahPoolConfig));
  3265. poolConfig.name = 'opfs-sahpool-cf37d5ff11';
  3266. let poolUtil;
  3267. await sqlite3.installOpfsSAHPoolVfs(poolConfig).then(p=>poolUtil=p);
  3268. T.assert(!!sqlite3.capi.sqlite3_vfs_find(poolConfig.name), "Expecting to find just-registered VFS");
  3269. try{
  3270. tryOne(false, "Emscripten filesystem");
  3271. tryOne(poolConfig.name);
  3272. tryOne('opfs');
  3273. }finally{
  3274. await poolUtil.removeVfs();
  3275. }
  3276. }
  3277. })
  3278. ;/*end of Bug Reports group*/;
  3279. ////////////////////////////////////////////////////////////////////////
  3280. log("Loading and initializing sqlite3 WASM module...");
  3281. if(0){
  3282. globalThis.sqlite3ApiConfig = {
  3283. debug: ()=>{},
  3284. log: ()=>{},
  3285. warn: ()=>{},
  3286. error: ()=>{}
  3287. }
  3288. }
  3289. //#ifnot target=es6-module
  3290. if(!globalThis.sqlite3InitModule && !isUIThread()){
  3291. /* Vanilla worker, as opposed to an ES6 module worker */
  3292. /*
  3293. If sqlite3.js is in a directory other than this script, in order
  3294. to get sqlite3.js to resolve sqlite3.wasm properly, we have to
  3295. explicitly tell it where sqlite3.js is being loaded from. We do
  3296. that by passing the `sqlite3.dir=theDirName` URL argument to
  3297. _this_ script. That URL argument will be seen by the JS/WASM
  3298. loader and it will adjust the sqlite3.wasm path accordingly. If
  3299. sqlite3.js/.wasm are in the same directory as this script then
  3300. that's not needed.
  3301. URL arguments passed as part of the filename via importScripts()
  3302. are simply lost, and such scripts see the globalThis.location of
  3303. _this_ script.
  3304. */
  3305. let sqlite3Js = 'sqlite3.js';
  3306. const urlParams = new URL(globalThis.location.href).searchParams;
  3307. if(urlParams.has('sqlite3.dir')){
  3308. sqlite3Js = urlParams.get('sqlite3.dir') + '/' + sqlite3Js;
  3309. }
  3310. importScripts(sqlite3Js);
  3311. }
  3312. //#endif
  3313. globalThis.sqlite3InitModule.__isUnderTest =
  3314. true /* disables certain API-internal cleanup so that we can
  3315. test internal APIs from here */;
  3316. globalThis.sqlite3InitModule({
  3317. print: log,
  3318. printErr: error
  3319. }).then(async function(sqlite3){
  3320. TestUtil.assert(!!sqlite3.util);
  3321. log("Done initializing WASM/JS bits. Running tests...");
  3322. sqlite3.config.warn("Installing sqlite3 bits as global S for local dev/test purposes.");
  3323. globalThis.S = sqlite3;
  3324. /*await sqlite3.installOpfsSAHPoolVfs(sahPoolConfig)
  3325. .then((u)=>log("Loaded",u.vfsName,"VFS"))
  3326. .catch(e=>{
  3327. log("Cannot install OpfsSAHPool.",e);
  3328. });*/
  3329. capi = sqlite3.capi;
  3330. wasm = sqlite3.wasm;
  3331. log("sqlite3 version:",capi.sqlite3_libversion(),
  3332. capi.sqlite3_sourceid());
  3333. if(wasm.bigIntEnabled){
  3334. log("BigInt/int64 support is enabled.");
  3335. }else{
  3336. logClass('warning',"BigInt/int64 support is disabled.");
  3337. }
  3338. if(haveWasmCTests()){
  3339. log("sqlite3__wasm_test_...() APIs are available.");
  3340. }else{
  3341. logClass('warning',"sqlite3__wasm_test_...() APIs unavailable.");
  3342. }
  3343. log("registered vfs list =",capi.sqlite3_js_vfs_list().join(', '));
  3344. TestUtil.runTests(sqlite3);
  3345. });
  3346. })(self);