@dockstat/sqlite-wrapper 1.1.2 → 1.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js DELETED
@@ -1,28 +0,0 @@
1
- // @bun
2
- import{Database as C}from"bun:sqlite";class P{state;constructor(M,X,Y){this.state={db:M,tableName:X,whereConditions:[],whereParams:[],regexConditions:[],jsonColumns:Y?.jsonColumns}}getDb(){return this.state.db}getTableName(){return this.state.tableName}buildWhereClause(){if(this.state.whereConditions.length===0)return["",[]];return[` WHERE ${this.state.whereConditions.join(" AND ")}`,this.state.whereParams.slice()]}hasRegexConditions(){return this.state.regexConditions.length>0}applyRegexFiltering(M){if(this.state.regexConditions.length===0)return M;return M.filter((X)=>this.state.regexConditions.every(({column:Y,regex:Z})=>{let _=X[String(Y)];if(_===null||_===void 0)return!1;return Z.test(String(_))}))}requireWhereClause(M){if(this.state.whereConditions.length===0&&this.state.regexConditions.length===0)throw new Error(`${M} operation requires at least one WHERE condition. Use where(), whereRaw(), whereIn(), whereOp(), or whereRgx() to add conditions.`)}quoteIdentifier(M){return`"${M.replace(/"/g,'""')}"`}resetWhereConditions(){this.state.whereConditions=[],this.state.whereParams=[],this.state.regexConditions=[]}transformRowFromDb(M){if(!this.state.jsonColumns||!M)return M;let X={...M};for(let Y of this.state.jsonColumns){let Z=String(Y);if(X[Z]&&typeof X[Z]==="string")try{X[Z]=JSON.parse(X[Z])}catch{}}return X}transformRowsFromDb(M){if(!this.state.jsonColumns)return M;return M.map((X)=>this.transformRowFromDb(X))}transformRowToDb(M){if(!this.state.jsonColumns||!M)return M;let X={...M};for(let Y of this.state.jsonColumns){let Z=String(Y);if(X[Z]!==void 0&&X[Z]!==null){if(typeof X[Z]==="object")X[Z]=JSON.stringify(X[Z])}}return X}}class R extends P{where(M){for(let[X,Y]of Object.entries(M))if(Y===null||Y===void 0)this.state.whereConditions.push(`${String(X)} IS NULL`);else this.state.whereConditions.push(`${String(X)} = ?`),this.state.whereParams.push(Y);return this}whereRgx(M){for(let[X,Y]of Object.entries(M))if(Y instanceof RegExp)this.state.regexConditions.push({column:X,regex:Y});else if(typeof Y==="string")this.state.regexConditions.push({column:X,regex:new RegExp(Y)});else if(Y!==null&&Y!==void 0)this.state.whereConditions.push(`${String(X)} = ?`),this.state.whereParams.push(Y);return this}whereExpr(M,X=[]){if(!M||typeof M!=="string")throw new Error("whereExpr: expr must be a non-empty string");if(this.state.whereConditions.push(`(${M})`),X.length)this.state.whereParams.push(...X);return this}whereRaw(M,X=[]){return this.whereExpr(M,X)}whereIn(M,X){if(!Array.isArray(X)||X.length===0)throw new Error("whereIn: values must be a non-empty array");let Y=X.map(()=>"?").join(", ");return this.state.whereConditions.push(`${String(M)} IN (${Y})`),this.state.whereParams.push(...X),this}whereNotIn(M,X){if(!Array.isArray(X)||X.length===0)throw new Error("whereNotIn: values must be a non-empty array");let Y=X.map(()=>"?").join(", ");return this.state.whereConditions.push(`${String(M)} NOT IN (${Y})`),this.state.whereParams.push(...X),this}whereOp(M,X,Y){let Z=(X??"").toUpperCase().trim();if(!["=","!=","<>","<","<=",">",">=","LIKE","GLOB","IS","IS NOT"].includes(Z))throw new Error(`whereOp: operator "${X}" not supported`);if((Y===null||Y===void 0)&&(Z==="="||Z==="IS"))return this.state.whereConditions.push(`${String(M)} IS NULL`),this;if((Y===null||Y===void 0)&&(Z==="!="||Z==="<>"||Z==="IS NOT"))return this.state.whereConditions.push(`${String(M)} IS NOT NULL`),this;return this.state.whereConditions.push(`${String(M)} ${Z} ?`),this.state.whereParams.push(Y),this}whereBetween(M,X,Y){return this.state.whereConditions.push(`${String(M)} BETWEEN ? AND ?`),this.state.whereParams.push(X,Y),this}whereNotBetween(M,X,Y){return this.state.whereConditions.push(`${String(M)} NOT BETWEEN ? AND ?`),this.state.whereParams.push(X,Y),this}whereNull(M){return this.state.whereConditions.push(`${String(M)} IS NULL`),this}whereNotNull(M){return this.state.whereConditions.push(`${String(M)} IS NOT NULL`),this}}class B extends R{selectedColumns=["*"];orderColumn;orderDirection="ASC";limitValue;offsetValue;select(M){return this.selectedColumns=M,this}orderBy(M){return this.orderColumn=M,this}desc(){return this.orderDirection="DESC",this}asc(){return this.orderDirection="ASC",this}limit(M){if(M<0)throw new Error("limit: amount must be non-negative");return this.limitValue=M,this}offset(M){if(M<0)throw new Error("offset: start must be non-negative");return this.offsetValue=M,this}buildSelectQuery(M=!0){let Y=`SELECT ${this.selectedColumns[0]==="*"?"*":this.selectedColumns.join(", ")} FROM ${this.quoteIdentifier(this.getTableName())}`,[Z,_]=this.buildWhereClause();if(Y+=Z,M&&!this.hasRegexConditions()){if(this.orderColumn)Y+=` ORDER BY ${String(this.orderColumn)} ${this.orderDirection}`;if(this.limitValue!==void 0)Y+=` LIMIT ${this.limitValue}`;if(this.offsetValue!==void 0)Y+=` OFFSET ${this.offsetValue}`}return[Y,_]}applyClientSideOperations(M){if(!this.hasRegexConditions())return M;let X=this.applyRegexFiltering(M);if(this.orderColumn){let Z=String(this.orderColumn);X.sort((_,$)=>{let H=_[Z],J=$[Z];if(H===J)return 0;if(H===null||H===void 0)return-1;if(J===null||J===void 0)return 1;if(H<J)return this.orderDirection==="ASC"?-1:1;return this.orderDirection==="ASC"?1:-1})}let Y=this.offsetValue??0;if(this.limitValue!==void 0)X=X.slice(Y,Y+this.limitValue);else if(Y>0)X=X.slice(Y);return X}all(){if(!this.hasRegexConditions()){let[_,$]=this.buildSelectQuery(!0),H=this.getDb().prepare(_).all(...$);return this.transformRowsFromDb(H)}let[M,X]=this.buildSelectQuery(!1),Y=this.getDb().prepare(M).all(...X),Z=this.transformRowsFromDb(Y);return this.applyClientSideOperations(Z)}get(){if(!this.hasRegexConditions()&&this.limitValue===void 0){let[X,Y]=this.buildSelectQuery(!0),Z=X.includes("LIMIT")?X:`${X} LIMIT 1`,_=this.getDb().prepare(Z).get(...Y);return _?this.transformRowFromDb(_):null}if(!this.hasRegexConditions()&&this.limitValue!==void 0){let[X,Y]=this.buildSelectQuery(!0),Z=this.getDb().prepare(X).get(...Y);return Z?this.transformRowFromDb(Z):null}return this.all()[0]??null}first(){let M=this.limitValue;this.limitValue=1;let X=this.get();return this.limitValue=M,X}count(){if(!this.hasRegexConditions()){let[M,X]=this.buildSelectQuery(!0),Y=M.replace(/SELECT (.+?) FROM/i,"SELECT COUNT(*) AS __count FROM");return this.getDb().prepare(Y).get(...X)?.__count??0}return this.all().length}exists(){if(!this.hasRegexConditions()){let[M,X]=this.buildSelectQuery(!0),Y=`SELECT EXISTS(${M}) AS __exists`,Z=this.getDb().prepare(Y).get(...X);return Boolean(Z?.__exists)}return this.count()>0}value(M){let X=this.first();return X?X[M]:null}pluck(M){return this.all().map((Y)=>Y[M])}}class L extends R{insert(M,X){let Z=(Array.isArray(M)?M:[M]).map((K)=>this.transformRowToDb(K));if(Z.length===0)throw new Error("insert: data cannot be empty");let _=new Set;for(let K of Z)for(let k of Object.keys(K))_.add(k);let $=Array.from(_);if($.length===0)throw new Error("insert: no columns to insert");let H="INSERT";if(X?.orIgnore)H="INSERT OR IGNORE";else if(X?.orReplace)H="INSERT OR REPLACE";else if(X?.orAbort)H="INSERT OR ABORT";else if(X?.orFail)H="INSERT OR FAIL";else if(X?.orRollback)H="INSERT OR ROLLBACK";let J=$.map((K)=>this.quoteIdentifier(K)).join(", "),F=$.map(()=>"?").join(", "),G=`${H} INTO ${this.quoteIdentifier(this.getTableName())} (${J}) VALUES (${F})`,A=this.getDb().prepare(G),W=0,z=0;for(let K of Z){let k=$.map((j)=>K[j]??null),O=A.run(...k);if(W+=O.changes,O.lastInsertRowid)z=Number(O.lastInsertRowid)}return{insertId:z,changes:W}}insertOrIgnore(M){return this.insert(M,{orIgnore:!0})}insertOrReplace(M){return this.insert(M,{orReplace:!0})}insertOrAbort(M){return this.insert(M,{orAbort:!0})}insertOrFail(M){return this.insert(M,{orFail:!0})}insertOrRollback(M){return this.insert(M,{orRollback:!0})}insertAndGet(M,X){let Y=this.insert(M,X);if(Y.changes===0)return null;if(Y.insertId>0)try{let Z=this.getDb().prepare(`SELECT * FROM ${this.quoteIdentifier(this.getTableName())} WHERE rowid = ?`).get(Y.insertId);return Z?this.transformRowFromDb(Z):null}catch{return null}return null}insertBatch(M,X){if(!Array.isArray(M)||M.length===0)throw new Error("insertBatch: rows must be a non-empty array");let Y=this.getDb();return Y.transaction((_)=>{let $=0,H=0,J=_.map((O)=>this.transformRowToDb(O)),F=new Set;for(let O of J)for(let j of Object.keys(O))F.add(j);let G=Array.from(F);if(G.length===0)throw new Error("insertBatch: no columns to insert");let A="INSERT";if(X?.orIgnore)A="INSERT OR IGNORE";else if(X?.orReplace)A="INSERT OR REPLACE";else if(X?.orAbort)A="INSERT OR ABORT";else if(X?.orFail)A="INSERT OR FAIL";else if(X?.orRollback)A="INSERT OR ROLLBACK";let W=G.map((O)=>this.quoteIdentifier(O)).join(", "),z=G.map(()=>"?").join(", "),K=`${A} INTO ${this.quoteIdentifier(this.getTableName())} (${W}) VALUES (${z})`,k=Y.prepare(K);for(let O of J){let j=G.map((I)=>O[I]??null),E=k.run(...j);if($+=E.changes,E.lastInsertRowid)H=Number(E.lastInsertRowid)}return{insertId:H,changes:$}})(M)}}class S extends B{update(M){this.requireWhereClause("UPDATE");let X=this.transformRowToDb(M),Y=Object.keys(X);if(Y.length===0)throw new Error("update: no columns to update");if(this.hasRegexConditions())return this.updateWithRegexConditions(X);let Z=Y.map((A)=>`${this.quoteIdentifier(A)} = ?`).join(", "),[_,$]=this.buildWhereClause(),H=`UPDATE ${this.quoteIdentifier(this.getTableName())} SET ${Z}${_}`,F=[...Y.map((A)=>X[A]),...$];return{changes:this.getDb().prepare(H).run(...F).changes}}updateWithRegexConditions(M){let[X,Y]=this.buildWhereClause(),Z=this.getDb().prepare(`SELECT rowid, * FROM ${this.quoteIdentifier(this.getTableName())}${X}`).all(...Y),_=this.applyRegexFiltering(Z);if(_.length===0)return{changes:0};let $=Object.keys(M),H=$.map((W)=>`${this.quoteIdentifier(W)} = ?`).join(", "),J=`UPDATE ${this.quoteIdentifier(this.getTableName())} SET ${H} WHERE rowid = ?`,F=this.getDb().prepare(J),G=0,A=$.map((W)=>M[W]);for(let W of _){let z=F.run(...A,W.rowid);G+=z.changes}return{changes:G}}upsert(M){let X=this.transformRowToDb(M),Y=Object.keys(X);if(Y.length===0)throw new Error("upsert: no columns to upsert");let Z=Y.map((F)=>this.quoteIdentifier(F)).join(", "),_=Y.map(()=>"?").join(", "),$=`INSERT OR REPLACE INTO ${this.quoteIdentifier(this.getTableName())} (${Z}) VALUES (${_})`,H=Y.map((F)=>X[F]??null);return{changes:this.getDb().prepare($).run(...H).changes}}increment(M,X=1){this.requireWhereClause("INCREMENT");let[Y,Z]=this.buildWhereClause(),_=`UPDATE ${this.quoteIdentifier(this.getTableName())} SET ${this.quoteIdentifier(String(M))} = ${this.quoteIdentifier(String(M))} + ?${Y}`;return{changes:this.getDb().prepare(_).run(X,...Z).changes}}decrement(M,X=1){return this.increment(M,-X)}updateAndGet(M){let X=this.all();if(this.update(M).changes===0)return[];return X}updateBatch(M){if(!Array.isArray(M)||M.length===0)throw new Error("updateBatch: updates must be a non-empty array");let X=this.getDb();return X.transaction((Z)=>{let _=0;for(let{where:$,data:H}of Z){let J=this.transformRowToDb(H),F=Object.keys(J);if(F.length===0)continue;let G=[],A=[];for(let[E,I]of Object.entries($))if(I===null||I===void 0)G.push(`${this.quoteIdentifier(E)} IS NULL`);else G.push(`${this.quoteIdentifier(E)} = ?`),A.push(I);if(G.length===0)throw new Error("updateBatch: each update must have WHERE conditions");let W=F.map((E)=>`${this.quoteIdentifier(E)} = ?`).join(", "),z=` WHERE ${G.join(" AND ")}`,K=`UPDATE ${this.quoteIdentifier(this.getTableName())} SET ${W}${z}`,O=[...F.map((E)=>J[E]??null),...A],j=X.prepare(K).run(...O);_+=j.changes}return{changes:_}})(M)}}class Q extends B{delete(){if(this.requireWhereClause("DELETE"),this.hasRegexConditions())return this.deleteWithRegexConditions();let[M,X]=this.buildWhereClause(),Y=`DELETE FROM ${this.quoteIdentifier(this.getTableName())}${M}`;return{changes:this.getDb().prepare(Y).run(...X).changes}}deleteWithRegexConditions(){let[M,X]=this.buildWhereClause(),Y=this.getDb().prepare(`SELECT rowid, * FROM ${this.quoteIdentifier(this.getTableName())}${M}`).all(...X),Z=this.applyRegexFiltering(Y);if(Z.length===0)return{changes:0};let _=`DELETE FROM ${this.quoteIdentifier(this.getTableName())} WHERE rowid = ?`,$=this.getDb().prepare(_),H=0;for(let J of Z){let F=$.run(J.rowid);H+=F.changes}return{changes:H}}deleteAndGet(){let M=this.all();if(this.delete().changes===0)return[];return M}softDelete(M="deleted_at",X=Math.floor(Date.now()/1000)){if(this.requireWhereClause("SOFT DELETE"),this.hasRegexConditions())return this.softDeleteWithRegexConditions(M,X);let[Y,Z]=this.buildWhereClause(),_=`UPDATE ${this.quoteIdentifier(this.getTableName())} SET ${this.quoteIdentifier(String(M))} = ?${Y}`;return{changes:this.getDb().prepare(_).run(X,...Z).changes}}softDeleteWithRegexConditions(M,X){let[Y,Z]=this.buildWhereClause(),_=this.getDb().prepare(`SELECT rowid, * FROM ${this.quoteIdentifier(this.getTableName())}${Y}`).all(...Z),$=this.applyRegexFiltering(_);if($.length===0)return{changes:0};let H=`UPDATE ${this.quoteIdentifier(this.getTableName())} SET ${this.quoteIdentifier(String(M))} = ? WHERE rowid = ?`,J=this.getDb().prepare(H),F=0;for(let G of $){let A=J.run(X,G.rowid);F+=A.changes}return{changes:F}}restore(M="deleted_at"){if(this.requireWhereClause("RESTORE"),this.hasRegexConditions())return this.restoreWithRegexConditions(M);let[X,Y]=this.buildWhereClause(),Z=`UPDATE ${this.quoteIdentifier(this.getTableName())} SET ${this.quoteIdentifier(String(M))} = NULL${X}`;return{changes:this.getDb().prepare(Z).run(...Y).changes}}restoreWithRegexConditions(M){let[X,Y]=this.buildWhereClause(),Z=this.getDb().prepare(`SELECT rowid, * FROM ${this.quoteIdentifier(this.getTableName())}${X}`).all(...Y),_=this.applyRegexFiltering(Z);if(_.length===0)return{changes:0};let $=`UPDATE ${this.quoteIdentifier(this.getTableName())} SET ${this.quoteIdentifier(String(M))} = NULL WHERE rowid = ?`,H=this.getDb().prepare($),J=0;for(let F of _){let G=H.run(F.rowid);J+=G.changes}return{changes:J}}deleteBatch(M){if(!Array.isArray(M)||M.length===0)throw new Error("deleteBatch: conditions must be a non-empty array");let X=this.getDb();return X.transaction((Z)=>{let _=0;for(let $ of Z){let H=[],J=[];for(let[W,z]of Object.entries($))if(z===null||z===void 0)H.push(`${this.quoteIdentifier(W)} IS NULL`);else H.push(`${this.quoteIdentifier(W)} = ?`),J.push(z);if(H.length===0)throw new Error("deleteBatch: each delete must have WHERE conditions");let F=` WHERE ${H.join(" AND ")}`,G=`DELETE FROM ${this.quoteIdentifier(this.getTableName())}${F}`,A=X.prepare(G).run(...J);_+=A.changes}return{changes:_}})(M)}truncate(){let M=`DELETE FROM ${this.quoteIdentifier(this.getTableName())}`;return{changes:this.getDb().prepare(M).run().changes}}deleteOlderThan(M,X){return this.whereOp(M,"<",X).delete()}deleteDuplicates(M){if(!Array.isArray(M)||M.length===0)throw new Error("deleteDuplicates: columns must be a non-empty array");let Y=M.map(($)=>String($)).map(($)=>this.quoteIdentifier($)).join(", "),Z=`
3
- DELETE FROM ${this.quoteIdentifier(this.getTableName())}
4
- WHERE rowid NOT IN (
5
- SELECT MIN(rowid)
6
- FROM ${this.quoteIdentifier(this.getTableName())}
7
- GROUP BY ${Y}
8
- )
9
- `;return{changes:this.getDb().prepare(Z).run().changes}}}class q{selectBuilder;insertBuilder;updateBuilder;deleteBuilder;constructor(M,X,Y){this.selectBuilder=new B(M,X,Y),this.insertBuilder=new L(M,X,Y),this.updateBuilder=new S(M,X,Y),this.deleteBuilder=new Q(M,X,Y),this.syncBuilderStates()}syncBuilderStates(){let M=this.selectBuilder.state;this.insertBuilder.state=M,this.updateBuilder.state=M,this.deleteBuilder.state=M}where(M){return this.selectBuilder.where(M),this}whereRgx(M){return this.selectBuilder.whereRgx(M),this}whereExpr(M,X=[]){return this.selectBuilder.whereExpr(M,X),this}whereRaw(M,X=[]){return this.selectBuilder.whereRaw(M,X),this}whereIn(M,X){return this.selectBuilder.whereIn(M,X),this}whereNotIn(M,X){return this.selectBuilder.whereNotIn(M,X),this}whereOp(M,X,Y){return this.selectBuilder.whereOp(M,X,Y),this}whereBetween(M,X,Y){return this.selectBuilder.whereBetween(M,X,Y),this}whereNotBetween(M,X,Y){return this.selectBuilder.whereNotBetween(M,X,Y),this}whereNull(M){return this.selectBuilder.whereNull(M),this}whereNotNull(M){return this.selectBuilder.whereNotNull(M),this}select(M){return this.selectBuilder.select(M),this}orderBy(M){return this.selectBuilder.orderBy(M),this}desc(){return this.selectBuilder.desc(),this}asc(){return this.selectBuilder.asc(),this}limit(M){return this.selectBuilder.limit(M),this}offset(M){return this.selectBuilder.offset(M),this}all(){return this.selectBuilder.all()}get(){return this.selectBuilder.get()}first(){return this.selectBuilder.first()}count(){return this.selectBuilder.count()}exists(){return this.selectBuilder.exists()}value(M){return this.selectBuilder.value(M)}pluck(M){return this.selectBuilder.pluck(M)}insert(M,X){return this.insertBuilder.insert(M,X)}insertOrIgnore(M){return this.insertBuilder.insertOrIgnore(M)}insertOrReplace(M){return this.insertBuilder.insertOrReplace(M)}insertOrAbort(M){return this.insertBuilder.insertOrAbort(M)}insertOrFail(M){return this.insertBuilder.insertOrFail(M)}insertOrRollback(M){return this.insertBuilder.insertOrRollback(M)}insertAndGet(M,X){return this.insertBuilder.insertAndGet(M,X)}insertBatch(M,X){return this.insertBuilder.insertBatch(M,X)}update(M){return this.updateBuilder.update(M)}upsert(M){return this.updateBuilder.upsert(M)}increment(M,X=1){return this.updateBuilder.increment(M,X)}decrement(M,X=1){return this.updateBuilder.decrement(M,X)}updateAndGet(M){return this.updateBuilder.updateAndGet(M)}updateBatch(M){return this.updateBuilder.updateBatch(M)}delete(){return this.deleteBuilder.delete()}deleteAndGet(){return this.deleteBuilder.deleteAndGet()}softDelete(M="deleted_at",X=Math.floor(Date.now()/1000)){return this.deleteBuilder.softDelete(M,X)}restore(M="deleted_at"){return this.deleteBuilder.restore(M)}deleteBatch(M){return this.deleteBuilder.deleteBatch(M)}truncate(){return this.deleteBuilder.truncate()}deleteOlderThan(M,X){return this.deleteBuilder.deleteOlderThan(M,X)}deleteDuplicates(M){return this.deleteBuilder.deleteDuplicates(M)}}var U={INTEGER:"INTEGER",TEXT:"TEXT",REAL:"REAL",BLOB:"BLOB",NUMERIC:"NUMERIC",INT:"INT",TINYINT:"TINYINT",SMALLINT:"SMALLINT",MEDIUMINT:"MEDIUMINT",BIGINT:"BIGINT",VARCHAR:"VARCHAR",CHAR:"CHAR",CHARACTER:"CHARACTER",NCHAR:"NCHAR",NVARCHAR:"NVARCHAR",CLOB:"CLOB",DOUBLE:"DOUBLE",FLOAT:"FLOAT",DECIMAL:"DECIMAL",DATE:"DATE",DATETIME:"DATETIME",TIMESTAMP:"TIMESTAMP",TIME:"TIME",BOOLEAN:"BOOLEAN",JSON:"JSON"},T={DATE:(M)=>`DATE(${M})`,TIME:(M)=>`TIME(${M})`,DATETIME:(M)=>`DATETIME(${M})`,JULIANDAY:(M)=>`JULIANDAY(${M})`,STRFTIME:(M,X)=>`STRFTIME('${M}', ${X})`,LENGTH:(M)=>`LENGTH(${M})`,LOWER:(M)=>`LOWER(${M})`,UPPER:(M)=>`UPPER(${M})`,TRIM:(M,X)=>X?`TRIM(${M}, '${X}')`:`TRIM(${M})`,LTRIM:(M,X)=>X?`LTRIM(${M}, '${X}')`:`LTRIM(${M})`,RTRIM:(M,X)=>X?`RTRIM(${M}, '${X}')`:`RTRIM(${M})`,SUBSTR:(M,X,Y)=>Y?`SUBSTR(${M}, ${X}, ${Y})`:`SUBSTR(${M}, ${X})`,SUBSTRING:(M,X,Y)=>Y?`SUBSTRING(${M}, ${X}, ${Y})`:`SUBSTRING(${M}, ${X})`,REPLACE:(M,X,Y)=>`REPLACE(${M}, '${X}', '${Y}')`,PRINTF:(M,...X)=>`PRINTF('${M}', ${X.join(", ")})`,ABS:(M)=>`ABS(${M})`,ROUND:(M,X)=>X!==void 0?`ROUND(${M}, ${X})`:`ROUND(${M})`,RANDOM:()=>"RANDOM()",MIN:(...M)=>`MIN(${M.join(", ")})`,MAX:(...M)=>`MAX(${M.join(", ")})`,CAST:(M,X)=>`CAST(${M} AS ${X})`,TYPEOF:(M)=>`TYPEOF(${M})`,COALESCE:(...M)=>`COALESCE(${M.join(", ")})`,IFNULL:(M,X)=>`IFNULL(${M}, ${X})`,NULLIF:(M,X)=>`NULLIF(${M}, ${X})`,IIF:(M,X,Y)=>`IIF(${M}, ${X}, ${Y})`,COUNT:(M)=>M?`COUNT(${M})`:"COUNT(*)",SUM:(M)=>`SUM(${M})`,AVG:(M)=>`AVG(${M})`,TOTAL:(M)=>`TOTAL(${M})`,GROUP_CONCAT:(M,X)=>X?`GROUP_CONCAT(${M}, '${X}')`:`GROUP_CONCAT(${M})`,JSON:(M)=>`JSON(${M})`,JSON_EXTRACT:(M,X)=>`JSON_EXTRACT(${M}, '${X}')`,JSON_TYPE:(M,X)=>X?`JSON_TYPE(${M}, '${X}')`:`JSON_TYPE(${M})`,JSON_VALID:(M)=>`JSON_VALID(${M})`,JSON_ARRAY:(...M)=>`JSON_ARRAY(${M.join(", ")})`,JSON_OBJECT:(...M)=>`JSON_OBJECT(${M.join(", ")})`},g={NULL:"NULL",CURRENT_TIME:"CURRENT_TIME",CURRENT_DATE:"CURRENT_DATE",CURRENT_TIMESTAMP:"CURRENT_TIMESTAMP",TRUE:"1",FALSE:"0",ROLLBACK:"ROLLBACK",ABORT:"ABORT",FAIL:"FAIL",IGNORE:"IGNORE",REPLACE:"REPLACE"},V=(M)=>({_type:"expression",expression:M}),b={integer:(M)=>({type:M?.size||U.INTEGER,...M}),text:(M)=>({type:M?.variant||U.TEXT,length:M?.length,...M}),real:(M)=>({type:M?.variant||U.REAL,...M}),blob:(M)=>({type:U.BLOB,...M}),numeric:(M)=>({type:M?.variant||U.NUMERIC,precision:M?.precision,scale:M?.scale,...M}),date:(M)=>({type:U.DATE,...M}),datetime:(M)=>({type:U.DATETIME,...M}),timestamp:(M)=>({type:M?.asText?U.TEXT:U.INTEGER,...M}),time:(M)=>({type:U.TIME,...M}),boolean:(M)=>({type:U.BOOLEAN,check:M?.check||"{{COLUMN}} IN (0, 1)",...M}),json:(M)=>({type:U.JSON,check:M?.validateJson?"JSON_VALID({{COLUMN}})":M?.check,...M}),varchar:(M,X)=>({type:U.VARCHAR,length:M,...X}),char:(M,X)=>({type:U.CHAR,length:M,...X}),id:(M)=>({type:U.INTEGER,primaryKey:!0,autoincrement:!0,notNull:!0,...M}),uuid:(M)=>({type:U.TEXT,length:36,default:M?.generateDefault?V("lower(hex(randomblob(4))) || '-' || lower(hex(randomblob(2))) || '-4' || substr(lower(hex(randomblob(2))),2) || '-' || substr('89ab',abs(random()) % 4 + 1, 1) || substr(lower(hex(randomblob(2))),2) || '-' || lower(hex(randomblob(6)))"):M?.default,...M}),createdAt:(M)=>({type:M?.asText?U.DATETIME:U.INTEGER,notNull:!0,default:M?.asText?V("datetime('now')"):V("strftime('%s', 'now')"),...M}),updatedAt:(M)=>({type:M?.asText?U.DATETIME:U.INTEGER,default:M?.asText?V("datetime('now')"):V("strftime('%s', 'now')"),...M}),foreignKey:(M,X="id",Y)=>({type:Y?.type||U.INTEGER,references:{table:M,column:X,onDelete:Y?.onDelete,onUpdate:Y?.onUpdate},...Y}),enum:(M,X)=>({type:U.TEXT,check:`{{COLUMN}} IN (${M.map((Y)=>`'${Y}'`).join(", ")})`,...X})},v={now:()=>V("datetime('now')"),currentTime:()=>V("time('now')"),currentDate:()=>V("date('now')"),currentTimestamp:()=>V("datetime('now')"),unixTimestamp:()=>V("strftime('%s', 'now')"),...T,raw:(M)=>V(M),null:()=>null,true:()=>1,false:()=>0};class N{db;constructor(M,X){if(this.db=new C(M),X?.pragmas)for(let[Y,Z]of X.pragmas)this.pragma(Y,Z);if(X?.loadExtensions)for(let Y of X.loadExtensions)this.loadExtension(Y)}table(M,X){return new q(this.db,M,X)}close(){this.db.close()}createTable(M,X,Y){let Z=Y?.temporary?"TEMPORARY ":"",_=Y?.ifNotExists?"IF NOT EXISTS ":"",$=Y?.withoutRowId?" WITHOUT ROWID":"",H=(W)=>`"${W.replace(/"/g,'""')}"`,J,F=[];if(typeof X==="string"){if(J=X.trim(),!J)throw new Error("Empty column definition string")}else if(this.isTableSchema(X)){let W=[];for(let[z,K]of Object.entries(X)){if(!z)continue;let k=this.buildColumnSQL(z,K);W.push(`${H(z)} ${k}`)}if(W.length===0)throw new Error("No columns provided");if(J=W.join(", "),Y?.constraints)F=this.buildTableConstraints(Y.constraints)}else{let W=[];for(let[z,K]of Object.entries(X)){if(!z)continue;let k=(K??"").trim();if(!k)throw new Error(`Missing SQL type/constraints for column "${z}"`);W.push(`${H(z)} ${k}`)}if(W.length===0)throw new Error("No columns provided");J=W.join(", ")}let G=[J,...F].join(", "),A=`CREATE ${Z}TABLE ${_}${H(M)} (${G})${$};`;if(this.db.exec(A),Y?.comment)this.setTableComment(M,Y.comment)}createIndex(M,X,Y,Z){let _=Z?.unique?"UNIQUE ":"",$=Z?.ifNotExists?"IF NOT EXISTS ":"",H=(G)=>`"${G.replace(/"/g,'""')}"`,J=Array.isArray(Y)?Y.map(H).join(", "):H(Y),F=`CREATE ${_}INDEX ${$}${H(M)} ON ${H(X)} (${J})`;if(Z?.where)F+=` WHERE ${Z.where}`;this.db.exec(`${F};`)}dropTable(M,X){let _=`DROP TABLE ${X?.ifExists?"IF EXISTS ":""}${(($)=>`"${$.replace(/"/g,'""')}"`)(M)};`;this.db.exec(_)}dropIndex(M,X){let _=`DROP INDEX ${X?.ifExists?"IF EXISTS ":""}${(($)=>`"${$.replace(/"/g,'""')}"`)(M)};`;this.db.exec(_)}isTableSchema(M){if(typeof M!=="object"||M===null)return!1;for(let[X,Y]of Object.entries(M))if(typeof Y==="object"&&Y!==null&&"type"in Y){let Z=Y.type;if(["INTEGER","TEXT","REAL","BLOB","NUMERIC","INT","TINYINT","SMALLINT","MEDIUMINT","BIGINT","VARCHAR","CHAR","CHARACTER","NCHAR","NVARCHAR","CLOB","DOUBLE","FLOAT","DECIMAL","DATE","DATETIME","TIMESTAMP","TIME","BOOLEAN","JSON"].includes(Z))return!0}return!1}buildColumnSQL(M,X){let Y=[],Z=X.type;if(X.length)Z+=`(${X.length})`;else if(X.precision!==void 0)if(X.scale!==void 0)Z+=`(${X.precision}, ${X.scale})`;else Z+=`(${X.precision})`;if(Y.push(Z),X.primaryKey)Y.push("PRIMARY KEY");if(X.autoincrement){if(!X.type.includes("INT")||!X.primaryKey)throw new Error(`AUTOINCREMENT can only be used with INTEGER PRIMARY KEY columns (column: ${M})`);Y.push("AUTOINCREMENT")}if(X.notNull&&!X.primaryKey)Y.push("NOT NULL");if(X.unique)Y.push("UNIQUE");if(X.default!==void 0)if(X.default===null)Y.push("DEFAULT NULL");else if(typeof X.default==="object"&&X.default._type==="expression")Y.push(`DEFAULT (${X.default.expression})`);else if(typeof X.default==="string")if(this.isSQLFunction(X.default))Y.push(`DEFAULT (${X.default})`);else Y.push(`DEFAULT '${X.default.replace(/'/g,"''")}'`);else if(typeof X.default==="boolean")Y.push(`DEFAULT ${X.default?1:0}`);else Y.push(`DEFAULT ${X.default}`);if(X.collate)Y.push(`COLLATE ${X.collate}`);if(X.check){let _=X.check.replace(/\{\{COLUMN\}\}/g,`"${M.replace(/"/g,'""')}"`);Y.push(`CHECK (${_})`)}if(X.references){let _=X.references,$=`REFERENCES "${_.table.replace(/"/g,'""')}"("${_.column.replace(/"/g,'""')}")`;if(_.onDelete)$+=` ON DELETE ${_.onDelete}`;if(_.onUpdate)$+=` ON UPDATE ${_.onUpdate}`;Y.push($)}if(X.generated){let _=X.generated.stored?"STORED":"VIRTUAL";Y.push(`GENERATED ALWAYS AS (${X.generated.expression}) ${_}`)}return Y.join(" ")}buildTableConstraints(M){let X=[];if(M.primaryKey&&M.primaryKey.length>0){let Y=M.primaryKey.map((Z)=>`"${Z.replace(/"/g,'""')}"`).join(", ");X.push(`PRIMARY KEY (${Y})`)}if(M.unique)if(Array.isArray(M.unique[0]))for(let Y of M.unique){let Z=Y.map((_)=>`"${_.replace(/"/g,'""')}"`).join(", ");X.push(`UNIQUE (${Z})`)}else{let Y=M.unique.map((Z)=>`"${Z.replace(/"/g,'""')}"`).join(", ");X.push(`UNIQUE (${Y})`)}if(M.check)for(let Y of M.check)X.push(`CHECK (${Y})`);if(M.foreignKeys)for(let Y of M.foreignKeys){let Z=Y.columns.map((H)=>`"${H.replace(/"/g,'""')}"`).join(", "),_=Y.references.columns.map((H)=>`"${H.replace(/"/g,'""')}"`).join(", "),$=`FOREIGN KEY (${Z}) REFERENCES "${Y.references.table.replace(/"/g,'""')}" (${_})`;if(Y.references.onDelete)$+=` ON DELETE ${Y.references.onDelete}`;if(Y.references.onUpdate)$+=` ON UPDATE ${Y.references.onUpdate}`;X.push($)}return X}isSQLFunction(M){return[/^\w+\s*\(/,/^(datetime|date|time|strftime|current_timestamp|current_date|current_time)/i,/^(random|abs|length|upper|lower|trim)/i,/^(coalesce|ifnull|nullif|iif)/i,/^(json|json_extract|json_valid)/i].some((Y)=>Y.test(M.trim()))}setTableComment(M,X){try{this.db.exec(`
10
- CREATE TABLE IF NOT EXISTS __table_metadata__ (
11
- table_name TEXT PRIMARY KEY,
12
- comment TEXT,
13
- created_at DATETIME DEFAULT CURRENT_TIMESTAMP
14
- )
15
- `),this.db.prepare(`
16
- INSERT OR REPLACE INTO __table_metadata__ (table_name, comment, created_at)
17
- VALUES (?, ?, CURRENT_TIMESTAMP)
18
- `).run(M,X)}catch(Y){console.warn(`Could not store table comment for ${M}:`,Y)}}getTableComment(M){try{return this.db.prepare(`
19
- SELECT comment FROM __table_metadata__ WHERE table_name = ?
20
- `).get(M)?.comment||null}catch(X){return null}}exec(M){this.db.exec(M)}prepare(M){return this.db.prepare(M)}transaction(M){return this.db.transaction(M)()}begin(M){let X=M?` ${M}`:"";this.db.exec(`BEGIN${X}`)}commit(){this.db.exec("COMMIT")}rollback(){this.db.exec("ROLLBACK")}savepoint(M){let X=`"${M.replace(/"/g,'""')}"`;this.db.exec(`SAVEPOINT ${X}`)}releaseSavepoint(M){let X=`"${M.replace(/"/g,'""')}"`;this.db.exec(`RELEASE SAVEPOINT ${X}`)}rollbackToSavepoint(M){let X=`"${M.replace(/"/g,'""')}"`;this.db.exec(`ROLLBACK TO SAVEPOINT ${X}`)}vacuum(){this.db.exec("VACUUM")}analyze(M){if(M){let X=`"${M.replace(/"/g,'""')}"`;this.db.exec(`ANALYZE ${X}`)}else this.db.exec("ANALYZE")}integrityCheck(){return this.db.prepare("PRAGMA integrity_check").all()}getSchema(){return this.db.prepare(`
21
- SELECT name, type, sql
22
- FROM sqlite_master
23
- WHERE type IN ('table', 'index', 'view', 'trigger')
24
- ORDER BY type, name
25
- `).all()}getTableInfo(M){return this.db.prepare(`PRAGMA table_info("${M.replace(/"/g,'""')}")`).all()}getForeignKeys(M){return this.db.prepare(`PRAGMA foreign_key_list("${M.replace(/"/g,'""')}")`).all()}getIndexes(M){return this.db.prepare(`PRAGMA index_list("${M.replace(/"/g,'""')}")`).all()}pragma(M,X){if(X!==void 0){this.db.exec(`PRAGMA ${M} = ${X}`);return}let Y=this.db.prepare(`PRAGMA ${M}`).get();return Object.values(Y)[0]}loadExtension(M){this.db.loadExtension(M)}getDb(){return this.db}}var XM=N;export{v as sql,V as defaultExpr,XM as default,b as column,U as SQLiteTypes,g as SQLiteKeywords,T as SQLiteFunctions,q as QueryBuilder,N as DB};
26
-
27
- //# debugId=7CE759C0B927428264756E2164756E21
28
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../index.ts", "../query-builder/base.ts", "../query-builder/where.ts", "../query-builder/select.ts", "../query-builder/insert.ts", "../query-builder/update.ts", "../query-builder/delete.ts", "../query-builder/index.ts", "../types.ts"],
  "sourcesContent": [
    "import { Database, type SQLQueryBindings } from \"bun:sqlite\";\nimport { QueryBuilder } from \"./query-builder/index\";\nimport type {\n  JsonColumnConfig,\n  TableSchema,\n  ColumnDefinition,\n  SQLiteType,\n  ColumnConstraints,\n  TableOptions,\n  DefaultExpression,\n  ForeignKeyAction,\n  TableConstraints,\n} from \"./types\";\n\n/**\n * Re-export all types and utilities\n */\nexport { QueryBuilder };\nexport type {\n  InsertResult,\n  UpdateResult,\n  DeleteResult,\n  InsertOptions,\n  ColumnNames,\n  WhereCondition,\n  RegexCondition,\n  JsonColumnConfig,\n  TableSchema,\n  ColumnDefinition,\n  SQLiteType,\n  ColumnConstraints,\n  TableOptions,\n  DefaultExpression,\n  ForeignKeyAction,\n  TableConstraints,\n} from \"./types\";\n\n// Re-export helper utilities\nexport {\n  column,\n  sql,\n  SQLiteTypes,\n  SQLiteFunctions,\n  SQLiteKeywords,\n  defaultExpr,\n} from \"./types\";\n\n/**\n * TypedSQLite — comprehensive wrapper around bun:sqlite `Database`.\n *\n * This class provides full type safety for SQLite operations with support for:\n * - All SQLite data types and variations\n * - Built-in SQL functions\n * - Complex constraints and relationships\n * - Generated columns\n * - Table-level constraints\n * - JSON column support\n * - And much more...\n */\nclass DB {\n  private db: Database;\n\n  /**\n   * Open or create a SQLite database at `path`.\n   *\n   * @param path - Path to the SQLite file (e.g. \"app.db\"). Use \":memory:\" for in-memory DB.\n   * @param options - Optional database configuration\n   */\n  constructor(\n    path: string,\n    options?: {\n      pragmas?: Array<[string, SQLQueryBindings]>;\n      loadExtensions?: string[];\n    },\n  ) {\n    this.db = new Database(path);\n\n    // Apply PRAGMA settings if provided\n    if (options?.pragmas) {\n      for (const [name, value] of options.pragmas) {\n        this.pragma(name, value);\n      }\n    }\n\n    // Load extensions if provided\n    if (options?.loadExtensions) {\n      for (const extensionPath of options.loadExtensions) {\n        this.loadExtension(extensionPath);\n      }\n    }\n  }\n\n  /**\n   * Get a typed QueryBuilder for a given table name.\n   * (Documentation remains the same as before...)\n   */\n  table<T extends Record<string, unknown>>(\n    tableName: string,\n    jsonConfig?: JsonColumnConfig<T>,\n  ): QueryBuilder<T> {\n    return new QueryBuilder<T>(this.db, tableName, jsonConfig);\n  }\n\n  /**\n   * Close the underlying SQLite database handle.\n   */\n  close(): void {\n    this.db.close();\n  }\n\n  /**\n   * Create a table with comprehensive type safety and feature support.\n   *\n   * Now supports all SQLite features:\n   *\n   * **Basic Usage:**\n   * ```ts\n   * import { column, sql } from \"./db\";\n   *\n   * db.createTable(\"users\", {\n   *   id: column.id(), // Auto-incrementing primary key\n   *   email: column.varchar(255, { unique: true, notNull: true }),\n   *   name: column.text({ notNull: true }),\n   *   age: column.integer({ check: 'age >= 0 AND age <= 150' }),\n   *   balance: column.numeric({ precision: 10, scale: 2, default: 0 }),\n   *   is_active: column.boolean({ default: sql.true() }),\n   *   metadata: column.json({ validateJson: true }),\n   *   created_at: column.createdAt(),\n   *   updated_at: column.updatedAt(),\n   * });\n   * ```\n   *\n   * **Advanced Features:**\n   * ```ts\n   * db.createTable(\"orders\", {\n   *   id: column.id(),\n   *   order_number: column.varchar(50, {\n   *     unique: true,\n   *     default: sql.raw(\"'ORD-' || strftime('%Y%m%d', 'now') || '-' || substr(hex(randomblob(4)), 1, 8)\")\n   *   }),\n   *   customer_id: column.foreignKey('users', 'id', {\n   *     onDelete: 'CASCADE',\n   *     onUpdate: 'RESTRICT'\n   *   }),\n   *   status: column.enum(['pending', 'paid', 'shipped', 'delivered'], {\n   *     default: 'pending'\n   *   }),\n   *   total: column.numeric({ precision: 10, scale: 2, notNull: true }),\n   *   // Generated column\n   *   display_total: {\n   *     type: 'TEXT',\n   *     generated: {\n   *       expression: \"printf('$%.2f', total)\",\n   *       stored: false // VIRTUAL column\n   *     }\n   *   },\n   * }, {\n   *   constraints: {\n   *     check: ['total >= 0'],\n   *     unique: [['customer_id', 'order_number']]\n   *   }\n   * });\n   * ```\n   *\n   * **Date/Time Columns:**\n   * ```ts\n   * db.createTable(\"events\", {\n   *   id: column.id(),\n   *   name: column.text({ notNull: true }),\n   *   event_date: column.date({ notNull: true }),\n   *   start_time: column.time(),\n   *   created_at: column.timestamp({ default: sql.unixTimestamp() }),\n   *   expires_at: column.datetime({\n   *     default: sql.raw(\"datetime('now', '+1 year')\")\n   *   }),\n   * });\n   * ```\n   *\n   * **JSON and Advanced Types:**\n   * ```ts\n   * db.createTable(\"products\", {\n   *   id: column.uuid({ generateDefault: true }), // UUID primary key\n   *   name: column.text({ notNull: true }),\n   *   price: column.real({ check: 'price > 0' }),\n   *   specifications: column.json({ validateJson: true }),\n   *   tags: column.text(), // JSON array\n   *   image_data: column.blob(),\n   *   search_vector: {\n   *     type: 'TEXT',\n   *     generated: {\n   *       expression: \"lower(name || ' ' || coalesce(json_extract(specifications, '$.description'), ''))\",\n   *       stored: true // STORED for indexing\n   *     }\n   *   }\n   * });\n   * ```\n   *\n   * @param tableName - Table name to create.\n   * @param columns - Column definitions (string, legacy object, or type-safe schema).\n   * @param options - Table options including constraints and metadata.\n   *\n   * @throws {Error} If column definitions are invalid or constraints conflict.\n   */\n  createTable<_T extends Record<string, unknown>>(\n    tableName: string,\n    columns: string | Record<string, string> | TableSchema,\n    options?: TableOptions,\n  ): void {\n    const temp = options?.temporary ? \"TEMPORARY \" : \"\";\n    const ifNot = options?.ifNotExists ? \"IF NOT EXISTS \" : \"\";\n    const withoutRowId = options?.withoutRowId ? \" WITHOUT ROWID\" : \"\";\n\n    const quoteIdent = (s: string) => `\"${s.replace(/\"/g, '\"\"')}\"`;\n\n    let columnDefs: string;\n    let tableConstraints: string[] = [];\n\n    if (typeof columns === \"string\") {\n      // Original string-based approach\n      columnDefs = columns.trim();\n      if (!columnDefs) {\n        throw new Error(\"Empty column definition string\");\n      }\n    } else if (this.isTableSchema(columns)) {\n      // New comprehensive type-safe approach\n      const parts: string[] = [];\n      for (const [colName, colDef] of Object.entries(columns)) {\n        if (!colName) continue;\n\n        const sqlDef = this.buildColumnSQL(colName, colDef);\n        parts.push(`${quoteIdent(colName)} ${sqlDef}`);\n      }\n\n      if (parts.length === 0) {\n        throw new Error(\"No columns provided\");\n      }\n      columnDefs = parts.join(\", \");\n\n      // Add table-level constraints\n      if (options?.constraints) {\n        tableConstraints = this.buildTableConstraints(options.constraints);\n      }\n    } else {\n      // Original object-based approach\n      const parts: string[] = [];\n      for (const [col, def] of Object.entries(columns)) {\n        if (!col) continue;\n\n        const defTrim = (def ?? \"\").trim();\n        if (!defTrim) {\n          throw new Error(`Missing SQL type/constraints for column \"${col}\"`);\n        }\n        parts.push(`${quoteIdent(col)} ${defTrim}`);\n      }\n\n      if (parts.length === 0) {\n        throw new Error(\"No columns provided\");\n      }\n      columnDefs = parts.join(\", \");\n    }\n\n    // Combine column definitions and table constraints\n    const allDefinitions = [columnDefs, ...tableConstraints].join(\", \");\n\n    const sql = `CREATE ${temp}TABLE ${ifNot}${quoteIdent(tableName)} (${allDefinitions})${withoutRowId};`;\n\n    this.db.exec(sql);\n\n    // Store table comment as metadata if provided\n    if (options?.comment) {\n      this.setTableComment(tableName, options.comment);\n    }\n  }\n\n  /**\n   * Create an index on a table\n   */\n  createIndex(\n    indexName: string,\n    tableName: string,\n    columns: string | string[],\n    options?: {\n      unique?: boolean;\n      ifNotExists?: boolean;\n      where?: string;\n      partial?: string;\n    },\n  ): void {\n    const unique = options?.unique ? \"UNIQUE \" : \"\";\n    const ifNot = options?.ifNotExists ? \"IF NOT EXISTS \" : \"\";\n    const quoteIdent = (s: string) => `\"${s.replace(/\"/g, '\"\"')}\"`;\n\n    const columnList = Array.isArray(columns)\n      ? columns.map(quoteIdent).join(\", \")\n      : quoteIdent(columns);\n\n    let sql = `CREATE ${unique}INDEX ${ifNot}${quoteIdent(indexName)} ON ${quoteIdent(tableName)} (${columnList})`;\n\n    if (options?.where) {\n      sql += ` WHERE ${options.where}`;\n    }\n\n    this.db.exec(`${sql};`);\n  }\n\n  /**\n   * Drop a table\n   */\n  dropTable(tableName: string, options?: { ifExists?: boolean }): void {\n    const ifExists = options?.ifExists ? \"IF EXISTS \" : \"\";\n    const quoteIdent = (s: string) => `\"${s.replace(/\"/g, '\"\"')}\"`;\n\n    const sql = `DROP TABLE ${ifExists}${quoteIdent(tableName)};`;\n    this.db.exec(sql);\n  }\n\n  /**\n   * Drop an index\n   */\n  dropIndex(indexName: string, options?: { ifExists?: boolean }): void {\n    const ifExists = options?.ifExists ? \"IF EXISTS \" : \"\";\n    const quoteIdent = (s: string) => `\"${s.replace(/\"/g, '\"\"')}\"`;\n\n    const sql = `DROP INDEX ${ifExists}${quoteIdent(indexName)};`;\n    this.db.exec(sql);\n  }\n\n  /**\n   * Type guard to check if columns definition is a TableSchema\n   */\n  private isTableSchema(columns: unknown): columns is TableSchema {\n    if (typeof columns !== \"object\" || columns === null) {\n      return false;\n    }\n\n    // Check if any value has a 'type' property with a valid SQLite type\n    for (const [_key, value] of Object.entries(columns)) {\n      if (typeof value === \"object\" && value !== null && \"type\" in value) {\n        const type = (value as { type: string }).type;\n        const validTypes = [\n          \"INTEGER\",\n          \"TEXT\",\n          \"REAL\",\n          \"BLOB\",\n          \"NUMERIC\",\n          \"INT\",\n          \"TINYINT\",\n          \"SMALLINT\",\n          \"MEDIUMINT\",\n          \"BIGINT\",\n          \"VARCHAR\",\n          \"CHAR\",\n          \"CHARACTER\",\n          \"NCHAR\",\n          \"NVARCHAR\",\n          \"CLOB\",\n          \"DOUBLE\",\n          \"FLOAT\",\n          \"DECIMAL\",\n          \"DATE\",\n          \"DATETIME\",\n          \"TIMESTAMP\",\n          \"TIME\",\n          \"BOOLEAN\",\n          \"JSON\",\n        ];\n        if (validTypes.includes(type)) {\n          return true;\n        }\n      }\n    }\n\n    return false;\n  }\n\n  /**\n   * Build SQL column definition from ColumnDefinition object\n   */\n  private buildColumnSQL(columnName: string, colDef: ColumnDefinition): string {\n    const parts: string[] = [];\n\n    // Add type with optional parameters\n    let typeStr = colDef.type;\n    if (colDef.length) {\n      typeStr += `(${colDef.length})`;\n    } else if (colDef.precision !== undefined) {\n      if (colDef.scale !== undefined) {\n        typeStr += `(${colDef.precision}, ${colDef.scale})`;\n      } else {\n        typeStr += `(${colDef.precision})`;\n      }\n    }\n    parts.push(typeStr);\n\n    // Add PRIMARY KEY (must come before AUTOINCREMENT)\n    if (colDef.primaryKey) {\n      parts.push(\"PRIMARY KEY\");\n    }\n\n    // Add AUTOINCREMENT (only valid with INTEGER PRIMARY KEY)\n    if (colDef.autoincrement) {\n      if (!colDef.type.includes(\"INT\") || !colDef.primaryKey) {\n        throw new Error(\n          `AUTOINCREMENT can only be used with INTEGER PRIMARY KEY columns (column: ${columnName})`,\n        );\n      }\n      parts.push(\"AUTOINCREMENT\");\n    }\n\n    // Add NOT NULL (but skip if PRIMARY KEY is already specified, as it's implicit)\n    if (colDef.notNull && !colDef.primaryKey) {\n      parts.push(\"NOT NULL\");\n    }\n\n    // Add UNIQUE\n    if (colDef.unique) {\n      parts.push(\"UNIQUE\");\n    }\n\n    // Add DEFAULT\n    if (colDef.default !== undefined) {\n      if (colDef.default === null) {\n        parts.push(\"DEFAULT NULL\");\n      } else if (\n        typeof colDef.default === \"object\" &&\n        colDef.default._type === \"expression\"\n      ) {\n        // Handle DefaultExpression\n        parts.push(`DEFAULT (${colDef.default.expression})`);\n      } else if (typeof colDef.default === \"string\") {\n        // Handle string defaults - check if it's a function call or literal\n        if (this.isSQLFunction(colDef.default)) {\n          parts.push(`DEFAULT (${colDef.default})`);\n        } else {\n          // Literal string value\n          parts.push(`DEFAULT '${colDef.default.replace(/'/g, \"''\")}'`);\n        }\n      } else if (typeof colDef.default === \"boolean\") {\n        parts.push(`DEFAULT ${colDef.default ? 1 : 0}`);\n      } else {\n        parts.push(`DEFAULT ${colDef.default}`);\n      }\n    }\n\n    // Add COLLATE\n    if (colDef.collate) {\n      parts.push(`COLLATE ${colDef.collate}`);\n    }\n\n    // Add CHECK constraint (replace placeholder with actual column name)\n    if (colDef.check) {\n      const checkConstraint = colDef.check.replace(\n        /\\{\\{COLUMN\\}\\}/g,\n        `\"${columnName.replace(/\"/g, '\"\"')}\"`,\n      );\n      parts.push(`CHECK (${checkConstraint})`);\n    }\n\n    // Add REFERENCES (foreign key)\n    if (colDef.references) {\n      const ref = colDef.references;\n      let refClause = `REFERENCES \"${ref.table.replace(/\"/g, '\"\"')}\"(\"${ref.column.replace(/\"/g, '\"\"')}\")`;\n\n      if (ref.onDelete) {\n        refClause += ` ON DELETE ${ref.onDelete}`;\n      }\n\n      if (ref.onUpdate) {\n        refClause += ` ON UPDATE ${ref.onUpdate}`;\n      }\n\n      parts.push(refClause);\n    }\n\n    // Add GENERATED column\n    if (colDef.generated) {\n      const storageType = colDef.generated.stored ? \"STORED\" : \"VIRTUAL\";\n      parts.push(\n        `GENERATED ALWAYS AS (${colDef.generated.expression}) ${storageType}`,\n      );\n    }\n\n    return parts.join(\" \");\n  }\n\n  /**\n   * Build table-level constraints\n   */\n  private buildTableConstraints(constraints: TableConstraints): string[] {\n    const parts: string[] = [];\n\n    // PRIMARY KEY constraint\n    if (constraints.primaryKey && constraints.primaryKey.length > 0) {\n      const columns = constraints.primaryKey\n        .map((col) => `\"${col.replace(/\"/g, '\"\"')}\"`)\n        .join(\", \");\n      parts.push(`PRIMARY KEY (${columns})`);\n    }\n\n    // UNIQUE constraints\n    if (constraints.unique) {\n      if (Array.isArray(constraints.unique[0])) {\n        // Multiple composite unique constraints\n        for (const uniqueGroup of constraints.unique as string[][]) {\n          const columns = uniqueGroup\n            .map((col) => `\"${col.replace(/\"/g, '\"\"')}\"`)\n            .join(\", \");\n          parts.push(`UNIQUE (${columns})`);\n        }\n      } else {\n        // Single unique constraint\n        const columns = (constraints.unique as string[])\n          .map((col) => `\"${col.replace(/\"/g, '\"\"')}\"`)\n          .join(\", \");\n        parts.push(`UNIQUE (${columns})`);\n      }\n    }\n\n    // CHECK constraints\n    if (constraints.check) {\n      for (const checkExpr of constraints.check) {\n        parts.push(`CHECK (${checkExpr})`);\n      }\n    }\n\n    // FOREIGN KEY constraints\n    if (constraints.foreignKeys) {\n      for (const fk of constraints.foreignKeys) {\n        const columns = fk.columns\n          .map((col) => `\"${col.replace(/\"/g, '\"\"')}\"`)\n          .join(\", \");\n        const refColumns = fk.references.columns\n          .map((col) => `\"${col.replace(/\"/g, '\"\"')}\"`)\n          .join(\", \");\n\n        let fkClause = `FOREIGN KEY (${columns}) REFERENCES \"${fk.references.table.replace(/\"/g, '\"\"')}\" (${refColumns})`;\n\n        if (fk.references.onDelete) {\n          fkClause += ` ON DELETE ${fk.references.onDelete}`;\n        }\n\n        if (fk.references.onUpdate) {\n          fkClause += ` ON UPDATE ${fk.references.onUpdate}`;\n        }\n\n        parts.push(fkClause);\n      }\n    }\n\n    return parts;\n  }\n\n  /**\n   * Check if a string looks like a SQL function call\n   */\n  private isSQLFunction(str: string): boolean {\n    // Simple heuristic: contains parentheses and common SQL function patterns\n    const functionPatterns = [\n      /^\\w+\\s*\\(/, // Function name followed by (\n      /^(datetime|date|time|strftime|current_timestamp|current_date|current_time)/i,\n      /^(random|abs|length|upper|lower|trim)/i,\n      /^(coalesce|ifnull|nullif|iif)/i,\n      /^(json|json_extract|json_valid)/i,\n    ];\n\n    return functionPatterns.some((pattern) => pattern.test(str.trim()));\n  }\n\n  /**\n   * Store table comment as metadata (using a system table if needed)\n   */\n  private setTableComment(tableName: string, comment: string): void {\n    // Create metadata table if it doesn't exist\n    try {\n      this.db.exec(`\n        CREATE TABLE IF NOT EXISTS __table_metadata__ (\n          table_name TEXT PRIMARY KEY,\n          comment TEXT,\n          created_at DATETIME DEFAULT CURRENT_TIMESTAMP\n        )\n      `);\n\n      // Insert or replace comment\n      const stmt = this.db.prepare(`\n        INSERT OR REPLACE INTO __table_metadata__ (table_name, comment, created_at)\n        VALUES (?, ?, CURRENT_TIMESTAMP)\n      `);\n      stmt.run(tableName, comment);\n    } catch (error) {\n      // Silently ignore if we can't create metadata table\n      console.warn(`Could not store table comment for ${tableName}:`, error);\n    }\n  }\n\n  /**\n   * Get table comment from metadata\n   */\n  getTableComment(tableName: string): string | null {\n    try {\n      const stmt = this.db.prepare(`\n        SELECT comment FROM __table_metadata__ WHERE table_name = ?\n      `);\n      const result = stmt.get(tableName) as { comment: string } | undefined;\n      return result?.comment || null;\n    } catch (_error) {\n      return null;\n    }\n  }\n\n  /**\n   * Execute a raw SQL statement\n   */\n  exec(sql: string): void {\n    this.db.exec(sql);\n  }\n\n  /**\n   * Prepare a SQL statement for repeated execution\n   */\n  prepare(sql: string) {\n    return this.db.prepare(sql);\n  }\n\n  /**\n   * Execute a transaction\n   */\n  transaction<T>(fn: () => T): T {\n    return this.db.transaction(fn)();\n  }\n\n  /**\n   * Begin a transaction manually\n   */\n  begin(mode?: \"DEFERRED\" | \"IMMEDIATE\" | \"EXCLUSIVE\"): void {\n    const modeStr = mode ? ` ${mode}` : \"\";\n    this.db.exec(`BEGIN${modeStr}`);\n  }\n\n  /**\n   * Commit a transaction\n   */\n  commit(): void {\n    this.db.exec(\"COMMIT\");\n  }\n\n  /**\n   * Rollback a transaction\n   */\n  rollback(): void {\n    this.db.exec(\"ROLLBACK\");\n  }\n\n  /**\n   * Create a savepoint\n   */\n  savepoint(name: string): void {\n    const quotedName = `\"${name.replace(/\"/g, '\"\"')}\"`;\n    this.db.exec(`SAVEPOINT ${quotedName}`);\n  }\n\n  /**\n   * Release a savepoint\n   */\n  releaseSavepoint(name: string): void {\n    const quotedName = `\"${name.replace(/\"/g, '\"\"')}\"`;\n    this.db.exec(`RELEASE SAVEPOINT ${quotedName}`);\n  }\n\n  /**\n   * Rollback to a savepoint\n   */\n  rollbackToSavepoint(name: string): void {\n    const quotedName = `\"${name.replace(/\"/g, '\"\"')}\"`;\n    this.db.exec(`ROLLBACK TO SAVEPOINT ${quotedName}`);\n  }\n\n  /**\n   * Vacuum the database (reclaim space and optimize)\n   */\n  vacuum(): void {\n    this.db.exec(\"VACUUM\");\n  }\n\n  /**\n   * Analyze the database (update statistics for query optimizer)\n   */\n  analyze(tableName?: string): void {\n    if (tableName) {\n      const quotedName = `\"${tableName.replace(/\"/g, '\"\"')}\"`;\n      this.db.exec(`ANALYZE ${quotedName}`);\n    } else {\n      this.db.exec(\"ANALYZE\");\n    }\n  }\n\n  /**\n   * Check database integrity\n   */\n  integrityCheck(): Array<{ integrity_check: string }> {\n    const stmt = this.db.prepare(\"PRAGMA integrity_check\");\n    return stmt.all() as Array<{ integrity_check: string }>;\n  }\n\n  /**\n   * Get database schema information\n   */\n  getSchema(): Array<{ name: string; type: string; sql: string }> {\n    const stmt = this.db.prepare(`\n      SELECT name, type, sql\n      FROM sqlite_master\n      WHERE type IN ('table', 'index', 'view', 'trigger')\n      ORDER BY type, name\n    `);\n    return stmt.all() as Array<{ name: string; type: string; sql: string }>;\n  }\n\n  /**\n   * Get table info (columns, types, constraints)\n   */\n  getTableInfo(tableName: string): Array<{\n    cid: number;\n    name: string;\n    type: string;\n    notnull: number;\n    dflt_value: SQLQueryBindings;\n    pk: number;\n  }> {\n    const stmt = this.db.prepare(\n      `PRAGMA table_info(\"${tableName.replace(/\"/g, '\"\"')}\")`,\n    );\n    return stmt.all() as Array<{\n      cid: number;\n      name: string;\n      type: string;\n      notnull: number;\n      dflt_value: SQLQueryBindings;\n      pk: number;\n    }>;\n  }\n\n  /**\n   * Get foreign key information for a table\n   */\n  getForeignKeys(tableName: string): Array<{\n    id: number;\n    seq: number;\n    table: string;\n    from: string;\n    to: string;\n    on_update: string;\n    on_delete: string;\n    match: string;\n  }> {\n    const stmt = this.db.prepare(\n      `PRAGMA foreign_key_list(\"${tableName.replace(/\"/g, '\"\"')}\")`,\n    );\n    return stmt.all() as Array<{\n      id: number;\n      seq: number;\n      table: string;\n      from: string;\n      to: string;\n      on_update: string;\n      on_delete: string;\n      match: string;\n    }>;\n  }\n\n  /**\n   * Get index information for a table\n   */\n  getIndexes(tableName: string): Array<{\n    name: string;\n    unique: number;\n    origin: string;\n    partial: number;\n  }> {\n    const stmt = this.db.prepare(\n      `PRAGMA index_list(\"${tableName.replace(/\"/g, '\"\"')}\")`,\n    );\n    return stmt.all() as Array<{\n      name: string;\n      unique: number;\n      origin: string;\n      partial: number;\n    }>;\n  }\n\n  /**\n   * Set or get a PRAGMA value.\n   *\n   * @param name - PRAGMA name (e.g., \"foreign_keys\", \"journal_mode\")\n   * @param value - Value to set (omit to get current value)\n   * @returns Current value when getting, undefined when setting\n   */\n  pragma(name: string, value?: SQLQueryBindings): SQLQueryBindings | undefined {\n    if (value !== undefined) {\n      this.db.exec(`PRAGMA ${name} = ${value}`);\n      return undefined;\n    }\n    const result = this.db.prepare(`PRAGMA ${name}`).get() as Record<\n      string,\n      SQLQueryBindings\n    >;\n    return Object.values(result)[0];\n  }\n\n  /**\n   * Load a SQLite extension.\n   *\n   * @param path - Absolute path to the compiled SQLite extension\n   */\n  loadExtension(path: string): void {\n    this.db.loadExtension(path);\n  }\n\n  /**\n   * Get direct access to the underlying SQLite database instance.\n   * Use this for advanced operations not covered by the wrapper.\n   *\n   * @returns The underlying Database instance\n   */\n  getDb(): Database {\n    return this.db;\n  }\n}\n\nexport { DB };\nexport default DB;\n",
    "import type { Database, SQLQueryBindings } from \"bun:sqlite\";\nimport type {\n  ColumnNames,\n  OrderDirection,\n  QueryBuilderState,\n  JsonColumnConfig,\n  DatabaseRowData,\n} from \"../types\";\n\n/**\n * Base QueryBuilder class that manages core state and shared functionality.\n * This class provides the foundation for all query operations.\n */\nexport abstract class BaseQueryBuilder<T extends Record<string, unknown>> {\n  protected state: QueryBuilderState<T>;\n\n  constructor(\n    db: Database,\n    tableName: string,\n    jsonConfig?: JsonColumnConfig<T>,\n  ) {\n    this.state = {\n      db,\n      tableName,\n      whereConditions: [],\n      whereParams: [],\n      regexConditions: [],\n      jsonColumns: jsonConfig?.jsonColumns,\n    };\n  }\n\n  /**\n   * Get the database instance\n   */\n  protected getDb(): Database {\n    return this.state.db;\n  }\n\n  /**\n   * Get the table name\n   */\n  protected getTableName(): string {\n    return this.state.tableName;\n  }\n\n  /**\n   * Build the WHERE clause portion of a SQL query.\n   * @returns Tuple of [whereClause, parameters] where whereClause includes \"WHERE\" prefix\n   */\n  protected buildWhereClause(): [string, SQLQueryBindings[]] {\n    if (this.state.whereConditions.length === 0) {\n      return [\"\", []];\n    }\n    return [\n      ` WHERE ${this.state.whereConditions.join(\" AND \")}`,\n      this.state.whereParams.slice(),\n    ];\n  }\n\n  /**\n   * Check if there are any regex conditions that require client-side filtering.\n   */\n  protected hasRegexConditions(): boolean {\n    return this.state.regexConditions.length > 0;\n  }\n\n  /**\n   * Apply client-side regex filtering to a set of rows.\n   * This is used when regex conditions are present.\n   */\n  protected applyRegexFiltering(rows: T[]): T[] {\n    if (this.state.regexConditions.length === 0) {\n      return rows;\n    }\n\n    return rows.filter((row) =>\n      this.state.regexConditions.every(({ column, regex }) => {\n        const value = row[String(column)];\n        if (value === null || value === undefined) return false;\n        return regex.test(String(value));\n      }),\n    );\n  }\n\n  /**\n   * Validate that WHERE conditions exist for operations that require them.\n   * Throws an error if no WHERE conditions are present.\n   */\n  protected requireWhereClause(operation: string): void {\n    if (\n      this.state.whereConditions.length === 0 &&\n      this.state.regexConditions.length === 0\n    ) {\n      throw new Error(\n        `${operation} operation requires at least one WHERE condition. Use where(), whereRaw(), whereIn(), whereOp(), or whereRgx() to add conditions.`,\n      );\n    }\n  }\n\n  /**\n   * Quote SQL identifiers to prevent injection and handle special characters.\n   */\n  protected quoteIdentifier(identifier: string): string {\n    return `\"${identifier.replace(/\"/g, '\"\"')}\"`;\n  }\n\n  /**\n   * Reset all WHERE conditions and parameters.\n   * Useful for reusing the same builder instance.\n   */\n  protected resetWhereConditions(): void {\n    this.state.whereConditions = [];\n    this.state.whereParams = [];\n    this.state.regexConditions = [];\n  }\n\n  /**\n   * Transform row data after fetching from database (deserialize JSON columns).\n   */\n  protected transformRowFromDb(row: unknown): T {\n    if (!this.state.jsonColumns || !row) return row as T;\n\n    const transformed = { ...row } as DatabaseRowData;\n    for (const column of this.state.jsonColumns) {\n      const columnKey = String(column);\n      if (\n        transformed[columnKey] &&\n        typeof transformed[columnKey] === \"string\"\n      ) {\n        try {\n          transformed[columnKey] = JSON.parse(transformed[columnKey] as string);\n        } catch {\n          // Keep original value if JSON parsing fails\n        }\n      }\n    }\n    return transformed as T;\n  }\n\n  /**\n   * Transform multiple rows after fetching from database.\n   */\n  protected transformRowsFromDb(rows: unknown[]): T[] {\n    if (!this.state.jsonColumns) return rows as T[];\n    return rows.map((row) => this.transformRowFromDb(row));\n  }\n\n  /**\n   * Transform row data before inserting/updating to database (serialize JSON columns).\n   */\n  protected transformRowToDb(row: Partial<T>): DatabaseRowData {\n    if (!this.state.jsonColumns || !row) return row as DatabaseRowData;\n\n    const transformed: DatabaseRowData = { ...row } as DatabaseRowData;\n    for (const column of this.state.jsonColumns) {\n      const columnKey = String(column);\n      if (\n        transformed[columnKey] !== undefined &&\n        transformed[columnKey] !== null\n      ) {\n        if (typeof transformed[columnKey] === \"object\") {\n          transformed[columnKey] = JSON.stringify(transformed[columnKey]);\n        }\n      }\n    }\n    return transformed;\n  }\n}\n",
    "import type { SQLQueryBindings } from \"bun:sqlite\";\nimport type { WhereCondition, RegexCondition } from \"../types\";\nimport { BaseQueryBuilder } from \"./base\";\n\n/**\n * Mixin class that adds WHERE-related functionality to the QueryBuilder.\n * This includes all conditional filtering methods.\n */\nexport class WhereQueryBuilder<\n  T extends Record<string, unknown>,\n> extends BaseQueryBuilder<T> {\n  /**\n   * Add simple equality conditions to the WHERE clause.\n   * Handles null values appropriately with IS NULL / IS NOT NULL.\n   *\n   * @param conditions - Object with column-value pairs for equality checks\n   * @returns this for method chaining\n   */\n  where(conditions: WhereCondition<T>): this {\n    for (const [column, value] of Object.entries(conditions)) {\n      if (value === null || value === undefined) {\n        this.state.whereConditions.push(`${String(column)} IS NULL`);\n      } else {\n        this.state.whereConditions.push(`${String(column)} = ?`);\n        this.state.whereParams.push(value);\n      }\n    }\n    return this;\n  }\n\n  /**\n   * Add regex conditions. Note: regex conditions are applied client-side\n   * after SQL execution due to Bun's SQLite limitations.\n   *\n   * @param conditions - Object with column-regex pairs\n   * @returns this for method chaining\n   */\n  whereRgx(conditions: RegexCondition<T>): this {\n    for (const [column, value] of Object.entries(conditions)) {\n      if (value instanceof RegExp) {\n        this.state.regexConditions.push({\n          column: column as keyof T,\n          regex: value,\n        });\n      } else if (typeof value === \"string\") {\n        this.state.regexConditions.push({\n          column: column as keyof T,\n          regex: new RegExp(value),\n        });\n      } else if (value !== null && value !== undefined) {\n        // Handle non-regex values as simple equality conditions\n        this.state.whereConditions.push(`${String(column)} = ?`);\n        this.state.whereParams.push(value);\n      }\n    }\n    return this;\n  }\n\n  /**\n   * Add a raw SQL WHERE fragment with parameter binding.\n   *\n   * @param expr - SQL fragment (without leading WHERE/AND), can use ? placeholders\n   * @param params - Values for the placeholders in order\n   * @returns this for method chaining\n   */\n  whereExpr(expr: string, params: SQLQueryBindings[] = []): this {\n    if (!expr || typeof expr !== \"string\") {\n      throw new Error(\"whereExpr: expr must be a non-empty string\");\n    }\n    // Wrap in parentheses to preserve grouping when combined with other clauses\n    this.state.whereConditions.push(`(${expr})`);\n    if (params.length) {\n      this.state.whereParams.push(...params);\n    }\n    return this;\n  }\n\n  /**\n   * Alias for whereExpr for compatibility\n   */\n  whereRaw(expr: string, params: SQLQueryBindings[] = []): this {\n    return this.whereExpr(expr, params);\n  }\n\n  /**\n   * Add an IN clause for the given column with proper parameter binding.\n   *\n   * @param column - Column name to check\n   * @param values - Non-empty array of values for the IN clause\n   * @returns this for method chaining\n   */\n  whereIn(column: keyof T, values: SQLQueryBindings[]): this {\n    if (!Array.isArray(values) || values.length === 0) {\n      throw new Error(\"whereIn: values must be a non-empty array\");\n    }\n    const placeholders = values.map(() => \"?\").join(\", \");\n    this.state.whereConditions.push(`${String(column)} IN (${placeholders})`);\n    this.state.whereParams.push(...values);\n    return this;\n  }\n\n  /**\n   * Add a NOT IN clause for the given column with proper parameter binding.\n   *\n   * @param column - Column name to check\n   * @param values - Non-empty array of values for the NOT IN clause\n   * @returns this for method chaining\n   */\n  whereNotIn(column: keyof T, values: SQLQueryBindings[]): this {\n    if (!Array.isArray(values) || values.length === 0) {\n      throw new Error(\"whereNotIn: values must be a non-empty array\");\n    }\n    const placeholders = values.map(() => \"?\").join(\", \");\n    this.state.whereConditions.push(\n      `${String(column)} NOT IN (${placeholders})`,\n    );\n    this.state.whereParams.push(...values);\n    return this;\n  }\n\n  /**\n   * Add a comparison operator condition with proper null handling.\n   * Supports: =, !=, <>, <, <=, >, >=, LIKE, GLOB, IS\n   *\n   * @param column - Column name\n   * @param op - Comparison operator\n   * @param value - Value to compare (handles null appropriately)\n   * @returns this for method chaining\n   */\n  whereOp(column: keyof T, op: string, value: SQLQueryBindings): this {\n    const normalizedOp = (op ?? \"\").toUpperCase().trim();\n    const allowed = [\n      \"=\",\n      \"!=\",\n      \"<>\",\n      \"<\",\n      \"<=\",\n      \">\",\n      \">=\",\n      \"LIKE\",\n      \"GLOB\",\n      \"IS\",\n      \"IS NOT\",\n    ];\n\n    if (!allowed.includes(normalizedOp)) {\n      throw new Error(`whereOp: operator \"${op}\" not supported`);\n    }\n\n    // Handle null special-casing for IS / IS NOT and equality operators\n    if (\n      (value === null || value === undefined) &&\n      (normalizedOp === \"=\" || normalizedOp === \"IS\")\n    ) {\n      this.state.whereConditions.push(`${String(column)} IS NULL`);\n      return this;\n    }\n\n    if (\n      (value === null || value === undefined) &&\n      (normalizedOp === \"!=\" ||\n        normalizedOp === \"<>\" ||\n        normalizedOp === \"IS NOT\")\n    ) {\n      this.state.whereConditions.push(`${String(column)} IS NOT NULL`);\n      return this;\n    }\n\n    // Normal param-bound condition\n    this.state.whereConditions.push(`${String(column)} ${normalizedOp} ?`);\n    this.state.whereParams.push(value);\n    return this;\n  }\n\n  /**\n   * Add a BETWEEN condition for the given column.\n   *\n   * @param column - Column name\n   * @param min - Minimum value (inclusive)\n   * @param max - Maximum value (inclusive)\n   * @returns this for method chaining\n   */\n  whereBetween(\n    column: keyof T,\n    min: SQLQueryBindings,\n    max: SQLQueryBindings,\n  ): this {\n    this.state.whereConditions.push(`${String(column)} BETWEEN ? AND ?`);\n    this.state.whereParams.push(min, max);\n    return this;\n  }\n\n  /**\n   * Add a NOT BETWEEN condition for the given column.\n   *\n   * @param column - Column name\n   * @param min - Minimum value (exclusive)\n   * @param max - Maximum value (exclusive)\n   * @returns this for method chaining\n   */\n  whereNotBetween(\n    column: keyof T,\n    min: SQLQueryBindings,\n    max: SQLQueryBindings,\n  ): this {\n    this.state.whereConditions.push(`${String(column)} NOT BETWEEN ? AND ?`);\n    this.state.whereParams.push(min, max);\n    return this;\n  }\n\n  /**\n   * Add an IS NULL condition for the given column.\n   *\n   * @param column - Column name\n   * @returns this for method chaining\n   */\n  whereNull(column: keyof T): this {\n    this.state.whereConditions.push(`${String(column)} IS NULL`);\n    return this;\n  }\n\n  /**\n   * Add an IS NOT NULL condition for the given column.\n   *\n   * @param column - Column name\n   * @returns this for method chaining\n   */\n  whereNotNull(column: keyof T): this {\n    this.state.whereConditions.push(`${String(column)} IS NOT NULL`);\n    return this;\n  }\n}\n",
    "import type { SQLQueryBindings } from \"bun:sqlite\";\nimport type { ColumnNames, OrderDirection } from \"../types\";\nimport { WhereQueryBuilder } from \"./where\";\n\n/**\n * Mixin class that adds SELECT-specific functionality to the QueryBuilder.\n * Handles column selection, ordering, limiting, and result execution methods.\n */\nexport class SelectQueryBuilder<\n  T extends Record<string, unknown>,\n> extends WhereQueryBuilder<T> {\n  private selectedColumns: ColumnNames<T> = [\"*\"];\n  private orderColumn?: keyof T;\n  private orderDirection: OrderDirection = \"ASC\";\n  private limitValue?: number;\n  private offsetValue?: number;\n\n  /**\n   * Specify which columns to select.\n   *\n   * @param columns - Array of column names or [\"*\"] for all columns\n   * @returns this for method chaining\n   */\n  select(columns: ColumnNames<T>): this {\n    this.selectedColumns = columns;\n    return this;\n  }\n\n  /**\n   * Add ORDER BY clause.\n   *\n   * @param column - Column name to order by\n   * @returns this for method chaining\n   */\n  orderBy(column: keyof T): this {\n    this.orderColumn = column;\n    return this;\n  }\n\n  /**\n   * Set order direction to descending.\n   *\n   * @returns this for method chaining\n   */\n  desc(): this {\n    this.orderDirection = \"DESC\";\n    return this;\n  }\n\n  /**\n   * Set order direction to ascending (default).\n   *\n   * @returns this for method chaining\n   */\n  asc(): this {\n    this.orderDirection = \"ASC\";\n    return this;\n  }\n\n  /**\n   * Add LIMIT clause.\n   *\n   * @param amount - Maximum number of rows to return\n   * @returns this for method chaining\n   */\n  limit(amount: number): this {\n    if (amount < 0) {\n      throw new Error(\"limit: amount must be non-negative\");\n    }\n    this.limitValue = amount;\n    return this;\n  }\n\n  /**\n   * Add OFFSET clause.\n   *\n   * @param start - Number of rows to skip\n   * @returns this for method chaining\n   */\n  offset(start: number): this {\n    if (start < 0) {\n      throw new Error(\"offset: start must be non-negative\");\n    }\n    this.offsetValue = start;\n    return this;\n  }\n\n  /**\n   * Build the complete SELECT query.\n   * If regex conditions exist, ORDER/LIMIT/OFFSET are not included in SQL\n   * as they will be applied client-side after regex filtering.\n   *\n   * @param includeOrderAndLimit - Whether to include ORDER/LIMIT/OFFSET in SQL\n   * @returns Tuple of [query, parameters]\n   */\n  private buildSelectQuery(\n    includeOrderAndLimit = true,\n  ): [string, SQLQueryBindings[]] {\n    const cols =\n      this.selectedColumns[0] === \"*\"\n        ? \"*\"\n        : (this.selectedColumns as string[]).join(\", \");\n\n    let query = `SELECT ${cols} FROM ${this.quoteIdentifier(this.getTableName())}`;\n\n    const [whereClause, whereParams] = this.buildWhereClause();\n    query += whereClause;\n\n    if (includeOrderAndLimit && !this.hasRegexConditions()) {\n      if (this.orderColumn) {\n        query += ` ORDER BY ${String(this.orderColumn)} ${this.orderDirection}`;\n      }\n\n      if (this.limitValue !== undefined) {\n        query += ` LIMIT ${this.limitValue}`;\n      }\n\n      if (this.offsetValue !== undefined) {\n        query += ` OFFSET ${this.offsetValue}`;\n      }\n    }\n\n    return [query, whereParams];\n  }\n\n  /**\n   * Apply JavaScript-based filtering, ordering, and pagination.\n   * Used when regex conditions require client-side processing.\n   *\n   * @param rows - Rows to process\n   * @returns Processed rows\n   */\n  private applyClientSideOperations(rows: T[]): T[] {\n    if (!this.hasRegexConditions()) return rows;\n\n    // Apply regex filters\n    let filtered = this.applyRegexFiltering(rows);\n\n    // Apply ordering in JavaScript\n    if (this.orderColumn) {\n      const col = String(this.orderColumn);\n      filtered.sort((a: T, b: T) => {\n        const va = a[col];\n        const vb = b[col];\n        if (va === vb) return 0;\n        if (va === null || va === undefined) return -1;\n        if (vb === null || vb === undefined) return 1;\n        if (va < vb) return this.orderDirection === \"ASC\" ? -1 : 1;\n        return this.orderDirection === \"ASC\" ? 1 : -1;\n      });\n    }\n\n    // Apply offset & limit in JavaScript\n    const start = this.offsetValue ?? 0;\n    if (this.limitValue !== undefined) {\n      filtered = filtered.slice(start, start + this.limitValue);\n    } else if (start > 0) {\n      filtered = filtered.slice(start);\n    }\n\n    return filtered;\n  }\n\n  /**\n   * Execute the query and return all matching rows.\n   *\n   * @returns Array of rows matching the query\n   */\n  all(): T[] {\n    if (!this.hasRegexConditions()) {\n      const [query, params] = this.buildSelectQuery(true);\n      const rows = this.getDb()\n        .prepare(query)\n        .all(...params) as T[];\n      return this.transformRowsFromDb(rows);\n    }\n\n    const [query, params] = this.buildSelectQuery(false);\n    const rows = this.getDb()\n      .prepare(query)\n      .all(...params) as T[];\n    const transformedRows = this.transformRowsFromDb(rows);\n    return this.applyClientSideOperations(transformedRows);\n  }\n\n  /**\n   * Execute the query and return the first matching row, or null if none found.\n   * If no explicit LIMIT is set, adds LIMIT 1 for efficiency.\n   *\n   * @returns First matching row or null\n   */\n  get(): T | null {\n    if (!this.hasRegexConditions() && this.limitValue === undefined) {\n      // No regex and no explicit limit, we can safely add LIMIT 1\n      const [query, params] = this.buildSelectQuery(true);\n      const q = query.includes(\"LIMIT\") ? query : `${query} LIMIT 1`;\n      const row = this.getDb()\n        .prepare(q)\n        .get(...params) as T | null;\n      return row ? this.transformRowFromDb(row) : null;\n    }\n\n    if (!this.hasRegexConditions() && this.limitValue !== undefined) {\n      // Limit is present; just use the query as-is\n      const [query, params] = this.buildSelectQuery(true);\n      const row = this.getDb()\n        .prepare(query)\n        .get(...params) as T | null;\n      return row ? this.transformRowFromDb(row) : null;\n    }\n\n    // Has regex conditions, need to process client-side\n    const results = this.all();\n    return results[0] ?? null;\n  }\n\n  /**\n   * Execute the query and return the first matching row, or null if none found.\n   * Always respects the semantics of returning the first row regardless of LIMIT.\n   *\n   * @returns First matching row or null\n   */\n  first(): T | null {\n    // Temporarily set limit to 1 but preserve previous value\n    const prevLimit = this.limitValue;\n    this.limitValue = 1;\n    const result = this.get();\n    this.limitValue = prevLimit;\n    return result;\n  }\n\n  /**\n   * Execute a COUNT query and return the number of matching rows.\n   * For regex conditions, this fetches all rows and counts client-side.\n   *\n   * @returns Number of matching rows\n   */\n  count(): number {\n    if (!this.hasRegexConditions()) {\n      // Safe to do COUNT(*) in SQL\n      const [baseQuery, params] = this.buildSelectQuery(true);\n      const countQuery = baseQuery.replace(\n        /SELECT (.+?) FROM/i,\n        \"SELECT COUNT(*) AS __count FROM\",\n      );\n      const result = this.getDb()\n        .prepare(countQuery)\n        .get(...params) as {\n        __count: number;\n      };\n      return result?.__count ?? 0;\n    }\n\n    // Has regex conditions, count client-side\n    return this.all().length;\n  }\n\n  /**\n   * Check if any rows match the current conditions.\n   *\n   * @returns true if at least one row matches, false otherwise\n   */\n  exists(): boolean {\n    if (!this.hasRegexConditions()) {\n      // Use EXISTS for efficiency\n      const [baseQuery, params] = this.buildSelectQuery(true);\n      const existsQuery = `SELECT EXISTS(${baseQuery}) AS __exists`;\n      const result = this.getDb()\n        .prepare(existsQuery)\n        .get(...params) as {\n        __exists: number;\n      };\n      return Boolean(result?.__exists);\n    }\n\n    // Has regex conditions, check client-side\n    return this.count() > 0;\n  }\n\n  /**\n   * Execute the query and return a single column value from the first row.\n   * Useful for getting a specific field value.\n   *\n   * @param column - Column name to extract the value from\n   * @returns The value of the specified column from the first row, or null\n   */\n  value<K extends keyof T>(column: K): T[K] | null {\n    const row = this.first();\n    return row ? row[column] : null;\n  }\n\n  /**\n   * Execute the query and return an array of values from a single column.\n   *\n   * @param column - Column name to extract values from\n   * @returns Array of values from the specified column\n   */\n  pluck<K extends keyof T>(column: K): T[K][] {\n    const rows = this.all();\n    return rows.map((row) => row[column]);\n  }\n}\n",
    "import type { SQLQueryBindings } from \"bun:sqlite\";\nimport type { InsertResult, InsertOptions } from \"../types\";\nimport { WhereQueryBuilder } from \"./where\";\n\n/**\n * Mixin class that adds INSERT functionality to the QueryBuilder.\n * Handles single and bulk insert operations with conflict resolution.\n */\nexport class InsertQueryBuilder<\n  T extends Record<string, unknown>,\n> extends WhereQueryBuilder<T> {\n  /**\n   * Insert a single row or multiple rows into the table.\n   *\n   * @param data - Single object or array of objects to insert\n   * @param options - Insert options (OR IGNORE, OR REPLACE, etc.)\n   * @returns Insert result with insertId and changes count\n   */\n  insert(\n    data: Partial<T> | Partial<T>[],\n    options?: InsertOptions,\n  ): InsertResult {\n    const rows = Array.isArray(data) ? data : [data];\n\n    // Transform rows to handle JSON serialization\n    const transformedRows = rows.map((row) => this.transformRowToDb(row));\n\n    if (transformedRows.length === 0) {\n      throw new Error(\"insert: data cannot be empty\");\n    }\n\n    // Get all unique columns from all rows\n    const allColumns = new Set<string>();\n    for (const row of transformedRows) {\n      for (const col of Object.keys(row)) {\n        allColumns.add(col);\n      }\n    }\n\n    const columns = Array.from(allColumns);\n    if (columns.length === 0) {\n      throw new Error(\"insert: no columns to insert\");\n    }\n\n    // Build INSERT statement with conflict resolution\n    let insertType = \"INSERT\";\n    if (options?.orIgnore) insertType = \"INSERT OR IGNORE\";\n    else if (options?.orReplace) insertType = \"INSERT OR REPLACE\";\n    else if (options?.orAbort) insertType = \"INSERT OR ABORT\";\n    else if (options?.orFail) insertType = \"INSERT OR FAIL\";\n    else if (options?.orRollback) insertType = \"INSERT OR ROLLBACK\";\n\n    const quotedColumns = columns\n      .map((col) => this.quoteIdentifier(col))\n      .join(\", \");\n    const placeholders = columns.map(() => \"?\").join(\", \");\n\n    const query = `${insertType} INTO ${this.quoteIdentifier(this.getTableName())} (${quotedColumns}) VALUES (${placeholders})`;\n    const stmt = this.getDb().prepare(query);\n\n    let totalChanges = 0;\n    let lastInsertId = 0;\n\n    // Execute for each row\n    for (const row of transformedRows) {\n      const values = columns.map(\n        (col) => row[col as keyof typeof row] ?? null,\n      ) as SQLQueryBindings[];\n      const result = stmt.run(...values);\n      totalChanges += result.changes;\n      if (result.lastInsertRowid) {\n        lastInsertId = Number(result.lastInsertRowid);\n      }\n    }\n\n    return {\n      insertId: lastInsertId,\n      changes: totalChanges,\n    };\n  }\n\n  /**\n   * Insert with OR IGNORE conflict resolution.\n   * Convenience method equivalent to insert(data, { orIgnore: true })\n   *\n   * @param data - Single object or array of objects to insert\n   * @returns Insert result with insertId and changes count\n   */\n  insertOrIgnore(data: Partial<T> | Partial<T>[]): InsertResult {\n    return this.insert(data, { orIgnore: true });\n  }\n\n  /**\n   * Insert with OR REPLACE conflict resolution.\n   * Convenience method equivalent to insert(data, { orReplace: true })\n   *\n   * @param data - Single object or array of objects to insert\n   * @returns Insert result with insertId and changes count\n   */\n  insertOrReplace(data: Partial<T> | Partial<T>[]): InsertResult {\n    return this.insert(data, { orReplace: true });\n  }\n\n  /**\n   * Insert with OR ABORT conflict resolution.\n   * This is the default behavior but provided for explicit usage.\n   *\n   * @param data - Single object or array of objects to insert\n   * @returns Insert result with insertId and changes count\n   */\n  insertOrAbort(data: Partial<T> | Partial<T>[]): InsertResult {\n    return this.insert(data, { orAbort: true });\n  }\n\n  /**\n   * Insert with OR FAIL conflict resolution.\n   *\n   * @param data - Single object or array of objects to insert\n   * @returns Insert result with insertId and changes count\n   */\n  insertOrFail(data: Partial<T> | Partial<T>[]): InsertResult {\n    return this.insert(data, { orFail: true });\n  }\n\n  /**\n   * Insert with OR ROLLBACK conflict resolution.\n   *\n   * @param data - Single object or array of objects to insert\n   * @returns Insert result with insertId and changes count\n   */\n  insertOrRollback(data: Partial<T> | Partial<T>[]): InsertResult {\n    return this.insert(data, { orRollback: true });\n  }\n\n  /**\n   * Insert and get the inserted row back.\n   * This is useful when you want to see the row with auto-generated fields.\n   *\n   * @param data - Single object to insert (bulk not supported for this method)\n   * @param options - Insert options\n   * @returns The inserted row with all fields, or null if insertion failed\n   */\n  insertAndGet(data: Partial<T>, options?: InsertOptions): T | null {\n    const result = this.insert(data, options);\n\n    if (result.changes === 0) {\n      return null;\n    }\n\n    // If we have an insertId, try to fetch the inserted row\n    if (result.insertId > 0) {\n      try {\n        const row = this.getDb()\n          .prepare(\n            `SELECT * FROM ${this.quoteIdentifier(this.getTableName())} WHERE rowid = ?`,\n          )\n          .get(result.insertId) as T | null;\n        return row ? this.transformRowFromDb(row) : null;\n      } catch {\n        // If fetching by rowid fails, return null\n        return null;\n      }\n    }\n\n    return null;\n  }\n\n  /**\n   * Batch insert with transaction support.\n   * This method wraps multiple inserts in a transaction for better performance\n   * and atomicity when inserting large amounts of data.\n   *\n   * @param rows - Array of objects to insert\n   * @param options - Insert options\n   * @returns Insert result with total changes\n   */\n  insertBatch(rows: Partial<T>[], options?: InsertOptions): InsertResult {\n    if (!Array.isArray(rows) || rows.length === 0) {\n      throw new Error(\"insertBatch: rows must be a non-empty array\");\n    }\n\n    const db = this.getDb();\n\n    // Use a transaction for batch operations\n    const transaction = db.transaction((rowsToInsert: Partial<T>[]) => {\n      let totalChanges = 0;\n      let lastInsertId = 0;\n\n      // Transform rows to handle JSON serialization\n      const transformedRows = rowsToInsert.map((row) =>\n        this.transformRowToDb(row),\n      );\n\n      // Get all unique columns from all rows\n      const allColumns = new Set<string>();\n      for (const row of transformedRows) {\n        for (const col of Object.keys(row)) {\n          allColumns.add(col);\n        }\n      }\n\n      const columns = Array.from(allColumns);\n      if (columns.length === 0) {\n        throw new Error(\"insertBatch: no columns to insert\");\n      }\n\n      // Build INSERT statement with conflict resolution\n      let insertType = \"INSERT\";\n      if (options?.orIgnore) insertType = \"INSERT OR IGNORE\";\n      else if (options?.orReplace) insertType = \"INSERT OR REPLACE\";\n      else if (options?.orAbort) insertType = \"INSERT OR ABORT\";\n      else if (options?.orFail) insertType = \"INSERT OR FAIL\";\n      else if (options?.orRollback) insertType = \"INSERT OR ROLLBACK\";\n\n      const quotedColumns = columns\n        .map((col) => this.quoteIdentifier(col))\n        .join(\", \");\n      const placeholders = columns.map(() => \"?\").join(\", \");\n\n      const query = `${insertType} INTO ${this.quoteIdentifier(this.getTableName())} (${quotedColumns}) VALUES (${placeholders})`;\n      const stmt = db.prepare(query);\n\n      for (const row of transformedRows) {\n        const values = columns.map(\n          (col) => row[col as keyof typeof row] ?? null,\n        ) as SQLQueryBindings[];\n        const result = stmt.run(...values);\n        totalChanges += result.changes;\n        if (result.lastInsertRowid) {\n          lastInsertId = Number(result.lastInsertRowid);\n        }\n      }\n\n      return { insertId: lastInsertId, changes: totalChanges };\n    });\n\n    return transaction(rows);\n  }\n}\n",
    "import type { SQLQueryBindings } from \"bun:sqlite\";\nimport type { UpdateResult } from \"../types\";\nimport { SelectQueryBuilder } from \"./select\";\n\n/**\n * Mixin class that adds UPDATE functionality to the QueryBuilder.\n * Handles safe update operations with mandatory WHERE conditions.\n */\nexport class UpdateQueryBuilder<\n  T extends Record<string, unknown>,\n> extends SelectQueryBuilder<T> {\n  /**\n   * Update rows matching the WHERE conditions with the provided data.\n   * Requires at least one WHERE condition to prevent accidental full table updates.\n   *\n   * @param data - Object with columns to update and their new values\n   * @returns Update result with changes count\n   */\n  update(data: Partial<T>): UpdateResult {\n    this.requireWhereClause(\"UPDATE\");\n\n    // Transform data to handle JSON serialization\n    const transformedData = this.transformRowToDb(data);\n    const updateColumns = Object.keys(transformedData);\n    if (updateColumns.length === 0) {\n      throw new Error(\"update: no columns to update\");\n    }\n\n    // Handle regex conditions by first fetching matching rows\n    if (this.hasRegexConditions()) {\n      return this.updateWithRegexConditions(transformedData);\n    }\n\n    // Build UPDATE statement\n    const setClause = updateColumns\n      .map((col) => `${this.quoteIdentifier(col)} = ?`)\n      .join(\", \");\n\n    const [whereClause, whereParams] = this.buildWhereClause();\n    const query = `UPDATE ${this.quoteIdentifier(this.getTableName())} SET ${setClause}${whereClause}`;\n\n    const updateValues = updateColumns.map((col) => transformedData[col]);\n    const allParams = [...updateValues, ...whereParams];\n\n    const result = this.getDb()\n      .prepare(query)\n      .run(...allParams);\n\n    return {\n      changes: result.changes,\n    };\n  }\n\n  /**\n   * Handle UPDATE operations when regex conditions are present.\n   * This requires client-side filtering and individual row updates.\n   */\n  private updateWithRegexConditions(\n    transformedData: Record<string, SQLQueryBindings>,\n  ): UpdateResult {\n    // First, get all rows matching SQL conditions (without regex)\n    const [selectQuery, selectParams] = this.buildWhereClause();\n    const candidateRows = this.getDb()\n      .prepare(\n        `SELECT rowid, * FROM ${this.quoteIdentifier(this.getTableName())}${selectQuery}`,\n      )\n      .all(...selectParams) as (T & { rowid: number })[];\n\n    // Apply regex filtering\n    const matchingRows = this.applyRegexFiltering(candidateRows);\n\n    if (matchingRows.length === 0) {\n      return { changes: 0 };\n    }\n\n    // Update each matching row by rowid\n    const updateColumns = Object.keys(transformedData);\n    const setClause = updateColumns\n      .map((col) => `${this.quoteIdentifier(col)} = ?`)\n      .join(\", \");\n\n    const updateQuery = `UPDATE ${this.quoteIdentifier(this.getTableName())} SET ${setClause} WHERE rowid = ?`;\n    const stmt = this.getDb().prepare(updateQuery);\n\n    let totalChanges = 0;\n    const updateValues = updateColumns.map((col) => transformedData[col]);\n\n    for (const row of matchingRows) {\n      const result = stmt.run(...updateValues, row.rowid as SQLQueryBindings);\n      totalChanges += result.changes;\n    }\n\n    return { changes: totalChanges };\n  }\n\n  /**\n   * Update or insert (upsert) functionality using INSERT OR REPLACE.\n   * This method attempts to update existing rows, and inserts new ones if they don't exist.\n   * Requires a unique constraint or primary key to work properly.\n   *\n   * @param data - Object with columns to upsert\n   * @returns Update result with changes count\n   */\n  upsert(data: Partial<T>): UpdateResult {\n    // Transform data to handle JSON serialization\n    const transformedData = this.transformRowToDb(data);\n    const columns = Object.keys(transformedData);\n    if (columns.length === 0) {\n      throw new Error(\"upsert: no columns to upsert\");\n    }\n\n    const quotedColumns = columns\n      .map((col) => this.quoteIdentifier(col))\n      .join(\", \");\n    const placeholders = columns.map(() => \"?\").join(\", \");\n\n    const query = `INSERT OR REPLACE INTO ${this.quoteIdentifier(this.getTableName())} (${quotedColumns}) VALUES (${placeholders})`;\n\n    const values = columns.map((col) => transformedData[col] ?? null);\n    const result = this.getDb()\n      .prepare(query)\n      .run(...values);\n\n    return {\n      changes: result.changes,\n    };\n  }\n\n  /**\n   * Increment a numeric column by a specified amount.\n   * This is more efficient than fetching, calculating, and updating.\n   *\n   * @param column - Column name to increment\n   * @param amount - Amount to increment by (defaults to 1)\n   * @returns Update result with changes count\n   */\n  increment(column: keyof T, amount = 1): UpdateResult {\n    this.requireWhereClause(\"INCREMENT\");\n\n    const [whereClause, whereParams] = this.buildWhereClause();\n    const query = `UPDATE ${this.quoteIdentifier(this.getTableName())} SET ${this.quoteIdentifier(String(column))} = ${this.quoteIdentifier(String(column))} + ?${whereClause}`;\n\n    const result = this.getDb()\n      .prepare(query)\n      .run(amount, ...whereParams);\n\n    return {\n      changes: result.changes,\n    };\n  }\n\n  /**\n   * Decrement a numeric column by a specified amount.\n   * This is more efficient than fetching, calculating, and updating.\n   *\n   * @param column - Column name to decrement\n   * @param amount - Amount to decrement by (defaults to 1)\n   * @returns Update result with changes count\n   */\n  decrement(column: keyof T, amount = 1): UpdateResult {\n    return this.increment(column, -amount);\n  }\n\n  /**\n   * Update and get the updated rows back.\n   * This is useful when you want to see the rows after the update.\n   *\n   * @param data - Object with columns to update and their new values\n   * @returns Array of updated rows\n   */\n  updateAndGet(data: Partial<T>): T[] {\n    // First, get the rows that will be updated\n    const rowsToUpdate = this.all();\n\n    // Perform the update\n    const updateResult = this.update(data);\n\n    if (updateResult.changes === 0) {\n      return [];\n    }\n\n    // For simplicity, return the originally matched rows\n    // In a real implementation, you might want to re-fetch the updated rows\n    return rowsToUpdate;\n  }\n\n  /**\n   * Batch update multiple rows with different values.\n   * This is more efficient than individual updates when updating many rows.\n   *\n   * @param updates - Array of objects, each containing update data and conditions\n   * @returns Update result with total changes count\n   */\n  updateBatch(\n    updates: Array<{ where: Partial<T>; data: Partial<T> }>,\n  ): UpdateResult {\n    if (!Array.isArray(updates) || updates.length === 0) {\n      throw new Error(\"updateBatch: updates must be a non-empty array\");\n    }\n\n    const db = this.getDb();\n\n    // Use a transaction for batch operations\n    const transaction = db.transaction(\n      (updatesToProcess: Array<{ where: Partial<T>; data: Partial<T> }>) => {\n        let totalChanges = 0;\n\n        for (const { where: whereData, data } of updatesToProcess) {\n          // Transform data to handle JSON serialization\n          const transformedUpdateData = this.transformRowToDb(data);\n          const updateColumns = Object.keys(transformedUpdateData);\n          if (updateColumns.length === 0) {\n            continue; // Skip empty updates\n          }\n\n          // Build WHERE conditions for this update\n          const whereConditions: string[] = [];\n          const whereParams: SQLQueryBindings[] = [];\n\n          for (const [column, value] of Object.entries(whereData)) {\n            if (value === null || value === undefined) {\n              whereConditions.push(`${this.quoteIdentifier(column)} IS NULL`);\n            } else {\n              whereConditions.push(`${this.quoteIdentifier(column)} = ?`);\n              whereParams.push(value);\n            }\n          }\n\n          if (whereConditions.length === 0) {\n            throw new Error(\n              \"updateBatch: each update must have WHERE conditions\",\n            );\n          }\n\n          // Build UPDATE statement\n          const setClause = updateColumns\n            .map((col) => `${this.quoteIdentifier(col)} = ?`)\n            .join(\", \");\n\n          const whereClause = ` WHERE ${whereConditions.join(\" AND \")}`;\n          const query = `UPDATE ${this.quoteIdentifier(this.getTableName())} SET ${setClause}${whereClause}`;\n\n          const updateValues = updateColumns.map(\n            (col) => transformedUpdateData[col] ?? null,\n          );\n          const allParams = [...updateValues, ...whereParams];\n\n          const result = db.prepare(query).run(...allParams);\n          totalChanges += result.changes;\n        }\n\n        return { changes: totalChanges };\n      },\n    );\n\n    return transaction(updates);\n  }\n}\n",
    "import type { SQLQueryBindings } from \"bun:sqlite\";\nimport type { DeleteResult } from \"../types\";\nimport { SelectQueryBuilder } from \"./select\";\n\n/**\n * Mixin class that adds DELETE functionality to the QueryBuilder.\n * Handles safe delete operations with mandatory WHERE conditions.\n */\nexport class DeleteQueryBuilder<\n  T extends Record<string, unknown>,\n> extends SelectQueryBuilder<T> {\n  /**\n   * Delete rows matching the WHERE conditions.\n   * Requires at least one WHERE condition to prevent accidental full table deletion.\n   *\n   * @returns Delete result with changes count\n   */\n  delete(): DeleteResult {\n    this.requireWhereClause(\"DELETE\");\n\n    // Handle regex conditions by first fetching matching rows\n    if (this.hasRegexConditions()) {\n      return this.deleteWithRegexConditions();\n    }\n\n    // Build DELETE statement\n    const [whereClause, whereParams] = this.buildWhereClause();\n    const query = `DELETE FROM ${this.quoteIdentifier(this.getTableName())}${whereClause}`;\n\n    const result = this.getDb()\n      .prepare(query)\n      .run(...whereParams);\n\n    return {\n      changes: result.changes,\n    };\n  }\n\n  /**\n   * Handle DELETE operations when regex conditions are present.\n   * This requires client-side filtering and individual row deletion.\n   */\n  private deleteWithRegexConditions(): DeleteResult {\n    // First, get all rows matching SQL conditions (without regex)\n    const [selectQuery, selectParams] = this.buildWhereClause();\n    const candidateRows = this.getDb()\n      .prepare(\n        `SELECT rowid, * FROM ${this.quoteIdentifier(this.getTableName())}${selectQuery}`,\n      )\n      .all(...selectParams) as (T & { rowid: number })[];\n\n    // Apply regex filtering\n    const matchingRows = this.applyRegexFiltering(candidateRows);\n\n    if (matchingRows.length === 0) {\n      return { changes: 0 };\n    }\n\n    // Delete each matching row by rowid\n    const deleteQuery = `DELETE FROM ${this.quoteIdentifier(this.getTableName())} WHERE rowid = ?`;\n    const stmt = this.getDb().prepare(deleteQuery);\n\n    let totalChanges = 0;\n    for (const row of matchingRows) {\n      const result = stmt.run(row.rowid as SQLQueryBindings);\n      totalChanges += result.changes;\n    }\n\n    return { changes: totalChanges };\n  }\n\n  /**\n   * Delete and get the deleted rows back.\n   * This is useful when you want to see what was deleted for logging or audit purposes.\n   *\n   * @returns Array of deleted rows\n   */\n  deleteAndGet(): T[] {\n    // First, get the rows that will be deleted\n    const rowsToDelete = this.all();\n\n    // Perform the delete\n    const deleteResult = this.delete();\n\n    if (deleteResult.changes === 0) {\n      return [];\n    }\n\n    // Return the rows that were deleted\n    return rowsToDelete;\n  }\n\n  /**\n   * Soft delete - mark rows as deleted instead of physically removing them.\n   * This requires a 'deleted_at' or similar column in your table schema.\n   *\n   * @param deletedColumn - Column name to mark as deleted (defaults to 'deleted_at')\n   * @param deletedValue - Value to set for the deleted marker (defaults to current timestamp)\n   * @returns Update result with changes count\n   */\n  softDelete(\n    deletedColumn: keyof T = \"deleted_at\" as keyof T,\n    deletedValue: SQLQueryBindings = Math.floor(Date.now() / 1000),\n  ): DeleteResult {\n    this.requireWhereClause(\"SOFT DELETE\");\n\n    // Handle regex conditions\n    if (this.hasRegexConditions()) {\n      return this.softDeleteWithRegexConditions(deletedColumn, deletedValue);\n    }\n\n    // Build UPDATE statement to mark as deleted\n    const [whereClause, whereParams] = this.buildWhereClause();\n    const query = `UPDATE ${this.quoteIdentifier(this.getTableName())} SET ${this.quoteIdentifier(String(deletedColumn))} = ?${whereClause}`;\n\n    const result = this.getDb()\n      .prepare(query)\n      .run(deletedValue, ...whereParams);\n\n    return {\n      changes: result.changes,\n    };\n  }\n\n  /**\n   * Handle soft delete operations when regex conditions are present.\n   */\n  private softDeleteWithRegexConditions(\n    deletedColumn: keyof T,\n    deletedValue: SQLQueryBindings,\n  ): DeleteResult {\n    // First, get all rows matching SQL conditions (without regex)\n    const [selectQuery, selectParams] = this.buildWhereClause();\n    const candidateRows = this.getDb()\n      .prepare(\n        `SELECT rowid, * FROM ${this.quoteIdentifier(this.getTableName())}${selectQuery}`,\n      )\n      .all(...selectParams) as (T & { rowid: number })[];\n\n    // Apply regex filtering\n    const matchingRows = this.applyRegexFiltering(candidateRows);\n\n    if (matchingRows.length === 0) {\n      return { changes: 0 };\n    }\n\n    // Soft delete each matching row by rowid\n    const updateQuery = `UPDATE ${this.quoteIdentifier(this.getTableName())} SET ${this.quoteIdentifier(String(deletedColumn))} = ? WHERE rowid = ?`;\n    const stmt = this.getDb().prepare(updateQuery);\n\n    let totalChanges = 0;\n    for (const row of matchingRows) {\n      const result = stmt.run(deletedValue, row.rowid as SQLQueryBindings);\n      totalChanges += result.changes;\n    }\n\n    return { changes: totalChanges };\n  }\n\n  /**\n   * Restore soft deleted rows by clearing the deleted marker.\n   *\n   * @param deletedColumn - Column name that marks rows as deleted\n   * @returns Update result with changes count\n   */\n  restore(deletedColumn: keyof T = \"deleted_at\" as keyof T): DeleteResult {\n    this.requireWhereClause(\"RESTORE\");\n\n    // Handle regex conditions\n    if (this.hasRegexConditions()) {\n      return this.restoreWithRegexConditions(deletedColumn);\n    }\n\n    // Build UPDATE statement to clear the deleted marker\n    const [whereClause, whereParams] = this.buildWhereClause();\n    const query = `UPDATE ${this.quoteIdentifier(this.getTableName())} SET ${this.quoteIdentifier(String(deletedColumn))} = NULL${whereClause}`;\n\n    const result = this.getDb()\n      .prepare(query)\n      .run(...whereParams);\n\n    return {\n      changes: result.changes,\n    };\n  }\n\n  /**\n   * Handle restore operations when regex conditions are present.\n   */\n  private restoreWithRegexConditions(deletedColumn: keyof T): DeleteResult {\n    // First, get all rows matching SQL conditions (without regex)\n    const [selectQuery, selectParams] = this.buildWhereClause();\n    const candidateRows = this.getDb()\n      .prepare(\n        `SELECT rowid, * FROM ${this.quoteIdentifier(this.getTableName())}${selectQuery}`,\n      )\n      .all(...selectParams) as (T & { rowid: number })[];\n\n    // Apply regex filtering\n    const matchingRows = this.applyRegexFiltering(candidateRows);\n\n    if (matchingRows.length === 0) {\n      return { changes: 0 };\n    }\n\n    // Restore each matching row by rowid\n    const updateQuery = `UPDATE ${this.quoteIdentifier(this.getTableName())} SET ${this.quoteIdentifier(String(deletedColumn))} = NULL WHERE rowid = ?`;\n    const stmt = this.getDb().prepare(updateQuery);\n\n    let totalChanges = 0;\n    for (const row of matchingRows) {\n      const result = stmt.run(row.rowid as SQLQueryBindings);\n      totalChanges += result.changes;\n    }\n\n    return { changes: totalChanges };\n  }\n\n  /**\n   * Batch delete multiple sets of rows based on different conditions.\n   * This is more efficient than individual deletes when removing many different row sets.\n   *\n   * @param conditions - Array of WHERE condition objects\n   * @returns Delete result with total changes count\n   */\n  deleteBatch(conditions: Array<Partial<T>>): DeleteResult {\n    if (!Array.isArray(conditions) || conditions.length === 0) {\n      throw new Error(\"deleteBatch: conditions must be a non-empty array\");\n    }\n\n    const db = this.getDb();\n\n    // Use a transaction for batch operations\n    const transaction = db.transaction(\n      (conditionsToProcess: Array<Partial<T>>) => {\n        let totalChanges = 0;\n\n        for (const whereData of conditionsToProcess) {\n          // Build WHERE conditions for this delete\n          const whereConditions: string[] = [];\n          const whereParams: SQLQueryBindings[] = [];\n\n          for (const [column, value] of Object.entries(whereData)) {\n            if (value === null || value === undefined) {\n              whereConditions.push(`${this.quoteIdentifier(column)} IS NULL`);\n            } else {\n              whereConditions.push(`${this.quoteIdentifier(column)} = ?`);\n              whereParams.push(value);\n            }\n          }\n\n          if (whereConditions.length === 0) {\n            throw new Error(\n              \"deleteBatch: each delete must have WHERE conditions\",\n            );\n          }\n\n          // Build DELETE statement\n          const whereClause = ` WHERE ${whereConditions.join(\" AND \")}`;\n          const query = `DELETE FROM ${this.quoteIdentifier(this.getTableName())}${whereClause}`;\n\n          const result = db.prepare(query).run(...whereParams);\n          totalChanges += result.changes;\n        }\n\n        return { changes: totalChanges };\n      },\n    );\n\n    return transaction(conditions);\n  }\n\n  /**\n   * Truncate the entire table (delete all rows).\n   * This bypasses the WHERE condition requirement since it's explicitly destructive.\n   * USE WITH EXTREME CAUTION!\n   *\n   * @returns Delete result with changes count\n   */\n  truncate(): DeleteResult {\n    const query = `DELETE FROM ${this.quoteIdentifier(this.getTableName())}`;\n    const result = this.getDb().prepare(query).run();\n\n    return {\n      changes: result.changes,\n    };\n  }\n\n  /**\n   * Delete rows older than a specified timestamp.\n   * Convenience method for common cleanup operations.\n   *\n   * @param timestampColumn - Column name containing the timestamp\n   * @param olderThan - Timestamp threshold (rows older than this will be deleted)\n   * @returns Delete result with changes count\n   */\n  deleteOlderThan(timestampColumn: keyof T, olderThan: number): DeleteResult {\n    return this.whereOp(timestampColumn, \"<\", olderThan).delete();\n  }\n\n  /**\n   * Delete duplicate rows based on specified columns, keeping only the first occurrence.\n   * This is useful for data cleanup operations.\n   *\n   * @param columns - Columns to check for duplicates\n   * @returns Delete result with changes count\n   */\n  deleteDuplicates(columns: Array<keyof T>): DeleteResult {\n    if (!Array.isArray(columns) || columns.length === 0) {\n      throw new Error(\"deleteDuplicates: columns must be a non-empty array\");\n    }\n\n    const columnNames = columns.map((col) => String(col));\n    const quotedColumns = columnNames\n      .map((col) => this.quoteIdentifier(col))\n      .join(\", \");\n\n    // Find duplicate rows, keeping the one with the minimum rowid\n    const query = `\n      DELETE FROM ${this.quoteIdentifier(this.getTableName())}\n      WHERE rowid NOT IN (\n        SELECT MIN(rowid)\n        FROM ${this.quoteIdentifier(this.getTableName())}\n        GROUP BY ${quotedColumns}\n      )\n    `;\n\n    const result = this.getDb().prepare(query).run();\n\n    return {\n      changes: result.changes,\n    };\n  }\n}\n",
    "import type { Database, SQLQueryBindings } from \"bun:sqlite\";\nimport type {\n  ColumnNames,\n  WhereCondition,\n  RegexCondition,\n  InsertResult,\n  UpdateResult,\n  DeleteResult,\n  InsertOptions,\n  JsonColumnConfig,\n  QueryBuilderState,\n} from \"../types\";\nimport { SelectQueryBuilder } from \"./select\";\nimport { InsertQueryBuilder } from \"./insert\";\nimport { UpdateQueryBuilder } from \"./update\";\nimport { DeleteQueryBuilder } from \"./delete\";\n\n/**\n * Main QueryBuilder class that combines all functionality using composition.\n * This class provides a unified interface for SELECT, INSERT, UPDATE, and DELETE operations.\n *\n * Each operation type is implemented in a separate module for better maintainability:\n * - SELECT: column selection, ordering, limiting, result execution\n * - INSERT: single/bulk inserts with conflict resolution\n * - UPDATE: safe updates with mandatory WHERE conditions\n * - DELETE: safe deletes with mandatory WHERE conditions\n * - WHERE: shared conditional logic across all operations\n */\nexport class QueryBuilder<T extends Record<string, unknown>> {\n  private selectBuilder: SelectQueryBuilder<T>;\n  private insertBuilder: InsertQueryBuilder<T>;\n  private updateBuilder: UpdateQueryBuilder<T>;\n  private deleteBuilder: DeleteQueryBuilder<T>;\n\n  constructor(\n    db: Database,\n    tableName: string,\n    jsonConfig?: JsonColumnConfig<T>,\n  ) {\n    // Create instances of each specialized builder\n    this.selectBuilder = new SelectQueryBuilder<T>(db, tableName, jsonConfig);\n    this.insertBuilder = new InsertQueryBuilder<T>(db, tableName, jsonConfig);\n    this.updateBuilder = new UpdateQueryBuilder<T>(db, tableName, jsonConfig);\n    this.deleteBuilder = new DeleteQueryBuilder<T>(db, tableName, jsonConfig);\n\n    // Ensure all builders share the same state for WHERE conditions\n    this.syncBuilderStates();\n  }\n\n  /**\n   * Synchronize the state between all builders so WHERE conditions are shared.\n   */\n  private syncBuilderStates(): void {\n    const masterState = (\n      this.selectBuilder as unknown as { state: QueryBuilderState<T> }\n    ).state;\n    (this.insertBuilder as unknown as { state: QueryBuilderState<T> }).state =\n      masterState;\n    (this.updateBuilder as unknown as { state: QueryBuilderState<T> }).state =\n      masterState;\n    (this.deleteBuilder as unknown as { state: QueryBuilderState<T> }).state =\n      masterState;\n  }\n\n  // ===== WHERE METHODS (delegated to selectBuilder) =====\n\n  /**\n   * Add simple equality conditions to the WHERE clause.\n   */\n  where(conditions: WhereCondition<T>): this {\n    this.selectBuilder.where(conditions);\n    return this;\n  }\n\n  /**\n   * Add regex conditions (applied client-side).\n   */\n  whereRgx(conditions: RegexCondition<T>): this {\n    this.selectBuilder.whereRgx(conditions);\n    return this;\n  }\n\n  /**\n   * Add a raw SQL WHERE fragment with parameter binding.\n   */\n  whereExpr(expr: string, params: SQLQueryBindings[] = []): this {\n    this.selectBuilder.whereExpr(expr, params);\n    return this;\n  }\n\n  /**\n   * Alias for whereExpr.\n   */\n  whereRaw(expr: string, params: SQLQueryBindings[] = []): this {\n    this.selectBuilder.whereRaw(expr, params);\n    return this;\n  }\n\n  /**\n   * Add an IN clause with proper parameter binding.\n   */\n  whereIn(column: keyof T, values: SQLQueryBindings[]): this {\n    this.selectBuilder.whereIn(column, values);\n    return this;\n  }\n\n  /**\n   * Add a NOT IN clause with proper parameter binding.\n   */\n  whereNotIn(column: keyof T, values: SQLQueryBindings[]): this {\n    this.selectBuilder.whereNotIn(column, values);\n    return this;\n  }\n\n  /**\n   * Add a comparison operator condition.\n   */\n  whereOp(column: keyof T, op: string, value: SQLQueryBindings): this {\n    this.selectBuilder.whereOp(column, op, value);\n    return this;\n  }\n\n  /**\n   * Add a BETWEEN condition.\n   */\n  whereBetween(\n    column: keyof T,\n    min: SQLQueryBindings,\n    max: SQLQueryBindings,\n  ): this {\n    this.selectBuilder.whereBetween(column, min, max);\n    return this;\n  }\n\n  /**\n   * Add a NOT BETWEEN condition.\n   */\n  whereNotBetween(\n    column: keyof T,\n    min: SQLQueryBindings,\n    max: SQLQueryBindings,\n  ): this {\n    this.selectBuilder.whereNotBetween(column, min, max);\n    return this;\n  }\n\n  /**\n   * Add an IS NULL condition.\n   */\n  whereNull(column: keyof T): this {\n    this.selectBuilder.whereNull(column);\n    return this;\n  }\n\n  /**\n   * Add an IS NOT NULL condition.\n   */\n  whereNotNull(column: keyof T): this {\n    this.selectBuilder.whereNotNull(column);\n    return this;\n  }\n\n  // ===== SELECT METHODS (delegated to selectBuilder) =====\n\n  /**\n   * Specify which columns to select.\n   */\n  select(columns: ColumnNames<T>): this {\n    this.selectBuilder.select(columns);\n    return this;\n  }\n\n  /**\n   * Add ORDER BY clause.\n   */\n  orderBy(column: keyof T): this {\n    this.selectBuilder.orderBy(column);\n    return this;\n  }\n\n  /**\n   * Set order direction to descending.\n   */\n  desc(): this {\n    this.selectBuilder.desc();\n    return this;\n  }\n\n  /**\n   * Set order direction to ascending.\n   */\n  asc(): this {\n    this.selectBuilder.asc();\n    return this;\n  }\n\n  /**\n   * Add LIMIT clause.\n   */\n  limit(amount: number): this {\n    this.selectBuilder.limit(amount);\n    return this;\n  }\n\n  /**\n   * Add OFFSET clause.\n   */\n  offset(start: number): this {\n    this.selectBuilder.offset(start);\n    return this;\n  }\n\n  // ===== SELECT EXECUTION METHODS (delegated to selectBuilder) =====\n\n  /**\n   * Execute the query and return all matching rows.\n   */\n  all(): T[] {\n    return this.selectBuilder.all();\n  }\n\n  /**\n   * Execute the query and return the first matching row, or null.\n   */\n  get(): T | null {\n    return this.selectBuilder.get();\n  }\n\n  /**\n   * Execute the query and return the first matching row, or null.\n   */\n  first(): T | null {\n    return this.selectBuilder.first();\n  }\n\n  /**\n   * Execute a COUNT query and return the number of matching rows.\n   */\n  count(): number {\n    return this.selectBuilder.count();\n  }\n\n  /**\n   * Check if any rows match the current conditions.\n   */\n  exists(): boolean {\n    return this.selectBuilder.exists();\n  }\n\n  /**\n   * Execute the query and return a single column value from the first row.\n   */\n  value<K extends keyof T>(column: K): T[K] | null {\n    return this.selectBuilder.value(column);\n  }\n\n  /**\n   * Execute the query and return an array of values from a single column.\n   */\n  pluck<K extends keyof T>(column: K): T[K][] {\n    return this.selectBuilder.pluck(column);\n  }\n\n  // ===== INSERT METHODS (delegated to insertBuilder) =====\n\n  /**\n   * Insert a single row or multiple rows into the table.\n   */\n  insert(\n    data: Partial<T> | Partial<T>[],\n    options?: InsertOptions,\n  ): InsertResult {\n    return this.insertBuilder.insert(data, options);\n  }\n\n  /**\n   * Insert with OR IGNORE conflict resolution.\n   */\n  insertOrIgnore(data: Partial<T> | Partial<T>[]): InsertResult {\n    return this.insertBuilder.insertOrIgnore(data);\n  }\n\n  /**\n   * Insert with OR REPLACE conflict resolution.\n   */\n  insertOrReplace(data: Partial<T> | Partial<T>[]): InsertResult {\n    return this.insertBuilder.insertOrReplace(data);\n  }\n\n  /**\n   * Insert with OR ABORT conflict resolution.\n   */\n  insertOrAbort(data: Partial<T> | Partial<T>[]): InsertResult {\n    return this.insertBuilder.insertOrAbort(data);\n  }\n\n  /**\n   * Insert with OR FAIL conflict resolution.\n   */\n  insertOrFail(data: Partial<T> | Partial<T>[]): InsertResult {\n    return this.insertBuilder.insertOrFail(data);\n  }\n\n  /**\n   * Insert with OR ROLLBACK conflict resolution.\n   */\n  insertOrRollback(data: Partial<T> | Partial<T>[]): InsertResult {\n    return this.insertBuilder.insertOrRollback(data);\n  }\n\n  /**\n   * Insert and get the inserted row back.\n   */\n  insertAndGet(data: Partial<T>, options?: InsertOptions): T | null {\n    return this.insertBuilder.insertAndGet(data, options);\n  }\n\n  /**\n   * Batch insert with transaction support.\n   */\n  insertBatch(rows: Partial<T>[], options?: InsertOptions): InsertResult {\n    return this.insertBuilder.insertBatch(rows, options);\n  }\n\n  // ===== UPDATE METHODS (delegated to updateBuilder) =====\n\n  /**\n   * Update rows matching the WHERE conditions.\n   */\n  update(data: Partial<T>): UpdateResult {\n    return this.updateBuilder.update(data);\n  }\n\n  /**\n   * Update or insert (upsert) using INSERT OR REPLACE.\n   */\n  upsert(data: Partial<T>): UpdateResult {\n    return this.updateBuilder.upsert(data);\n  }\n\n  /**\n   * Increment a numeric column by a specified amount.\n   */\n  increment(column: keyof T, amount = 1): UpdateResult {\n    return this.updateBuilder.increment(column, amount);\n  }\n\n  /**\n   * Decrement a numeric column by a specified amount.\n   */\n  decrement(column: keyof T, amount = 1): UpdateResult {\n    return this.updateBuilder.decrement(column, amount);\n  }\n\n  /**\n   * Update and get the updated rows back.\n   */\n  updateAndGet(data: Partial<T>): T[] {\n    return this.updateBuilder.updateAndGet(data);\n  }\n\n  /**\n   * Batch update multiple rows with different values.\n   */\n  updateBatch(\n    updates: Array<{ where: Partial<T>; data: Partial<T> }>,\n  ): UpdateResult {\n    return this.updateBuilder.updateBatch(updates);\n  }\n\n  // ===== DELETE METHODS (delegated to deleteBuilder) =====\n\n  /**\n   * Delete rows matching the WHERE conditions.\n   */\n  delete(): DeleteResult {\n    return this.deleteBuilder.delete();\n  }\n\n  /**\n   * Delete and get the deleted rows back.\n   */\n  deleteAndGet(): T[] {\n    return this.deleteBuilder.deleteAndGet();\n  }\n\n  /**\n   * Soft delete - mark rows as deleted instead of physically removing them.\n   */\n  softDelete(\n    deletedColumn: keyof T = \"deleted_at\" as keyof T,\n    deletedValue: SQLQueryBindings = Math.floor(Date.now() / 1000),\n  ): DeleteResult {\n    return this.deleteBuilder.softDelete(deletedColumn, deletedValue);\n  }\n\n  /**\n   * Restore soft deleted rows.\n   */\n  restore(deletedColumn: keyof T = \"deleted_at\" as keyof T): DeleteResult {\n    return this.deleteBuilder.restore(deletedColumn);\n  }\n\n  /**\n   * Batch delete multiple sets of rows.\n   */\n  deleteBatch(conditions: Array<Partial<T>>): DeleteResult {\n    return this.deleteBuilder.deleteBatch(conditions);\n  }\n\n  /**\n   * Truncate the entire table (delete all rows).\n   */\n  truncate(): DeleteResult {\n    return this.deleteBuilder.truncate();\n  }\n\n  /**\n   * Delete rows older than a specified timestamp.\n   */\n  deleteOlderThan(timestampColumn: keyof T, olderThan: number): DeleteResult {\n    return this.deleteBuilder.deleteOlderThan(timestampColumn, olderThan);\n  }\n\n  /**\n   * Delete duplicate rows based on specified columns.\n   */\n  deleteDuplicates(columns: Array<keyof T>): DeleteResult {\n    return this.deleteBuilder.deleteDuplicates(columns);\n  }\n}\n",
    "// Enhanced types.ts with comprehensive SQLite support\nimport type { Database, SQLQueryBindings } from \"bun:sqlite\";\n\n/**\n * All SQLite data types including affinity types\n */\nexport const SQLiteTypes = {\n  // Standard SQLite types\n  INTEGER: \"INTEGER\" as const,\n  TEXT: \"TEXT\" as const,\n  REAL: \"REAL\" as const,\n  BLOB: \"BLOB\" as const,\n  NUMERIC: \"NUMERIC\" as const,\n\n  // Common type aliases and variations\n  INT: \"INT\" as const,\n  TINYINT: \"TINYINT\" as const,\n  SMALLINT: \"SMALLINT\" as const,\n  MEDIUMINT: \"MEDIUMINT\" as const,\n  BIGINT: \"BIGINT\" as const,\n\n  // Text variations\n  VARCHAR: \"VARCHAR\" as const,\n  CHAR: \"CHAR\" as const,\n  CHARACTER: \"CHARACTER\" as const,\n  NCHAR: \"NCHAR\" as const,\n  NVARCHAR: \"NVARCHAR\" as const,\n  CLOB: \"CLOB\" as const,\n\n  // Numeric variations\n  DOUBLE: \"DOUBLE\" as const,\n  FLOAT: \"FLOAT\" as const,\n  DECIMAL: \"DECIMAL\" as const,\n\n  // Date/Time (stored as TEXT, INTEGER, or REAL)\n  DATE: \"DATE\" as const,\n  DATETIME: \"DATETIME\" as const,\n  TIMESTAMP: \"TIMESTAMP\" as const,\n  TIME: \"TIME\" as const,\n\n  // Boolean (stored as INTEGER)\n  BOOLEAN: \"BOOLEAN\" as const,\n\n  // JSON (stored as TEXT)\n  JSON: \"JSON\" as const,\n} as const;\n\nexport type SQLiteType = (typeof SQLiteTypes)[keyof typeof SQLiteTypes];\n\n/**\n * SQLite built-in scalar functions\n */\nexport const SQLiteFunctions = {\n  // Date/Time functions\n  DATE: (expr: string) => `DATE(${expr})`,\n  TIME: (expr: string) => `TIME(${expr})`,\n  DATETIME: (expr: string) => `DATETIME(${expr})`,\n  JULIANDAY: (expr: string) => `JULIANDAY(${expr})`,\n  STRFTIME: (format: string, expr: string) => `STRFTIME('${format}', ${expr})`,\n\n  // String functions\n  LENGTH: (expr: string) => `LENGTH(${expr})`,\n  LOWER: (expr: string) => `LOWER(${expr})`,\n  UPPER: (expr: string) => `UPPER(${expr})`,\n  TRIM: (expr: string, chars?: string) =>\n    chars ? `TRIM(${expr}, '${chars}')` : `TRIM(${expr})`,\n  LTRIM: (expr: string, chars?: string) =>\n    chars ? `LTRIM(${expr}, '${chars}')` : `LTRIM(${expr})`,\n  RTRIM: (expr: string, chars?: string) =>\n    chars ? `RTRIM(${expr}, '${chars}')` : `RTRIM(${expr})`,\n  SUBSTR: (expr: string, start: number, length?: number) =>\n    length\n      ? `SUBSTR(${expr}, ${start}, ${length})`\n      : `SUBSTR(${expr}, ${start})`,\n  SUBSTRING: (expr: string, start: number, length?: number) =>\n    length\n      ? `SUBSTRING(${expr}, ${start}, ${length})`\n      : `SUBSTRING(${expr}, ${start})`,\n  REPLACE: (expr: string, old: string, replacement: string) =>\n    `REPLACE(${expr}, '${old}', '${replacement}')`,\n  PRINTF: (format: string, ...args: string[]) =>\n    `PRINTF('${format}', ${args.join(\", \")})`,\n\n  // Math functions\n  ABS: (expr: string) => `ABS(${expr})`,\n  ROUND: (expr: string, digits?: number) =>\n    digits !== undefined ? `ROUND(${expr}, ${digits})` : `ROUND(${expr})`,\n  RANDOM: () => \"RANDOM()\",\n  MIN: (...exprs: string[]) => `MIN(${exprs.join(\", \")})`,\n  MAX: (...exprs: string[]) => `MAX(${exprs.join(\", \")})`,\n\n  // Type conversion\n  CAST: (expr: string, type: string) => `CAST(${expr} AS ${type})`,\n  TYPEOF: (expr: string) => `TYPEOF(${expr})`,\n\n  // Conditional\n  COALESCE: (...exprs: string[]) => `COALESCE(${exprs.join(\", \")})`,\n  IFNULL: (expr: string, replacement: string) =>\n    `IFNULL(${expr}, ${replacement})`,\n  NULLIF: (expr1: string, expr2: string) => `NULLIF(${expr1}, ${expr2})`,\n  IIF: (condition: string, trueValue: string, falseValue: string) =>\n    `IIF(${condition}, ${trueValue}, ${falseValue})`,\n\n  // Aggregate functions (for use in expressions)\n  COUNT: (expr?: string) => (expr ? `COUNT(${expr})` : \"COUNT(*)\"),\n  SUM: (expr: string) => `SUM(${expr})`,\n  AVG: (expr: string) => `AVG(${expr})`,\n  TOTAL: (expr: string) => `TOTAL(${expr})`,\n  GROUP_CONCAT: (expr: string, separator?: string) =>\n    separator\n      ? `GROUP_CONCAT(${expr}, '${separator}')`\n      : `GROUP_CONCAT(${expr})`,\n\n  // JSON functions (SQLite 3.45+)\n  JSON: (expr: string) => `JSON(${expr})`,\n  JSON_EXTRACT: (json: string, path: string) =>\n    `JSON_EXTRACT(${json}, '${path}')`,\n  JSON_TYPE: (json: string, path?: string) =>\n    path ? `JSON_TYPE(${json}, '${path}')` : `JSON_TYPE(${json})`,\n  JSON_VALID: (expr: string) => `JSON_VALID(${expr})`,\n  JSON_ARRAY: (...values: string[]) => `JSON_ARRAY(${values.join(\", \")})`,\n  JSON_OBJECT: (...pairs: string[]) => `JSON_OBJECT(${pairs.join(\", \")})`,\n} as const;\n\n/**\n * SQLite keywords and special values\n */\nexport const SQLiteKeywords = {\n  // Special values\n  NULL: \"NULL\" as const,\n  CURRENT_TIME: \"CURRENT_TIME\" as const,\n  CURRENT_DATE: \"CURRENT_DATE\" as const,\n  CURRENT_TIMESTAMP: \"CURRENT_TIMESTAMP\" as const,\n\n  // Boolean values (as integers)\n  TRUE: \"1\" as const,\n  FALSE: \"0\" as const,\n\n  // Conflict resolution\n  ROLLBACK: \"ROLLBACK\" as const,\n  ABORT: \"ABORT\" as const,\n  FAIL: \"FAIL\" as const,\n  IGNORE: \"IGNORE\" as const,\n  REPLACE: \"REPLACE\" as const,\n} as const;\n\n/**\n * Foreign key actions\n */\nexport type ForeignKeyAction =\n  | \"CASCADE\"\n  | \"SET NULL\"\n  | \"RESTRICT\"\n  | \"NO ACTION\"\n  | \"SET DEFAULT\";\n\n/**\n * Column constraint options with comprehensive support\n */\nexport interface ColumnConstraints {\n  /** Column is PRIMARY KEY */\n  primaryKey?: boolean;\n  /** Column has AUTOINCREMENT (only valid with INTEGER PRIMARY KEY) */\n  autoincrement?: boolean;\n  /** Column is NOT NULL */\n  notNull?: boolean;\n  /** Column has UNIQUE constraint */\n  unique?: boolean;\n  /** Default value - can be literal value, function call, or keyword */\n  default?: string | number | boolean | null | DefaultExpression;\n  /** Custom check constraint */\n  check?: string;\n  /** Collation sequence */\n  collate?: \"BINARY\" | \"NOCASE\" | \"RTRIM\" | string;\n  /** References another table (foreign key) */\n  references?: {\n    table: string;\n    column: string;\n    onDelete?: ForeignKeyAction;\n    onUpdate?: ForeignKeyAction;\n  };\n  /** Generated column expression (GENERATED ALWAYS AS) */\n  generated?: {\n    expression: string;\n    stored?: boolean; // true for STORED, false/undefined for VIRTUAL\n  };\n  /** Column comment (stored as metadata, not in schema) */\n  comment?: string;\n}\n\n/**\n * Type-safe default value expressions\n */\nexport type DefaultExpression = {\n  _type: \"expression\";\n  expression: string;\n};\n\n/**\n * Helper to create default expressions\n */\nexport const defaultExpr = (expression: string): DefaultExpression => ({\n  _type: \"expression\",\n  expression,\n});\n\n/**\n * Type-safe column definition\n */\nexport interface ColumnDefinition extends ColumnConstraints {\n  /** SQLite data type */\n  type: SQLiteType;\n  /** Optional type parameters (e.g., VARCHAR(255)) */\n  length?: number;\n  /** Precision for DECIMAL/NUMERIC types */\n  precision?: number;\n  /** Scale for DECIMAL/NUMERIC types */\n  scale?: number;\n}\n\n/**\n * Type-safe table schema definition\n */\nexport type TableSchema = Record<string, ColumnDefinition>;\n\n/**\n * Table constraint types\n */\nexport interface TableConstraints {\n  /** PRIMARY KEY constraint on multiple columns */\n  primaryKey?: string[];\n  /** UNIQUE constraints */\n  unique?: string[] | string[][];\n  /** CHECK constraints */\n  check?: string[];\n  /** FOREIGN KEY constraints */\n  foreignKeys?: Array<{\n    columns: string[];\n    references: {\n      table: string;\n      columns: string[];\n      onDelete?: ForeignKeyAction;\n      onUpdate?: ForeignKeyAction;\n    };\n  }>;\n}\n\n/**\n * Enhanced table options\n */\nexport interface TableOptions {\n  /** Add IF NOT EXISTS clause */\n  ifNotExists?: boolean;\n  /** Create WITHOUT ROWID table */\n  withoutRowId?: boolean;\n  /** Table constraints */\n  constraints?: TableConstraints;\n  /** Temporary table */\n  temporary?: boolean;\n  /** Table comment */\n  comment?: string;\n}\n\n/**\n * Comprehensive column helper functions with full type support\n */\nexport const column = {\n  /**\n   * Create an INTEGER column with optional size specification\n   */\n  integer: (\n    constraints?: ColumnConstraints & {\n      size?: \"TINYINT\" | \"SMALLINT\" | \"MEDIUMINT\" | \"BIGINT\";\n    },\n  ): ColumnDefinition => ({\n    type: constraints?.size || SQLiteTypes.INTEGER,\n    ...constraints,\n  }),\n\n  /**\n   * Create a TEXT column with optional length\n   */\n  text: (\n    constraints?: ColumnConstraints & {\n      length?: number;\n      variant?: \"VARCHAR\" | \"CHAR\" | \"CLOB\" | \"NCHAR\" | \"NVARCHAR\";\n    },\n  ): ColumnDefinition => ({\n    type: constraints?.variant || SQLiteTypes.TEXT,\n    length: constraints?.length,\n    ...constraints,\n  }),\n\n  /**\n   * Create a REAL column\n   */\n  real: (\n    constraints?: ColumnConstraints & { variant?: \"DOUBLE\" | \"FLOAT\" },\n  ): ColumnDefinition => ({\n    type: constraints?.variant || SQLiteTypes.REAL,\n    ...constraints,\n  }),\n\n  /**\n   * Create a BLOB column\n   */\n  blob: (constraints?: ColumnConstraints): ColumnDefinition => ({\n    type: SQLiteTypes.BLOB,\n    ...constraints,\n  }),\n\n  /**\n   * Create a NUMERIC/DECIMAL column with precision and scale\n   */\n  numeric: (\n    constraints?: ColumnConstraints & {\n      precision?: number;\n      scale?: number;\n      variant?: \"DECIMAL\";\n    },\n  ): ColumnDefinition => ({\n    type: constraints?.variant || SQLiteTypes.NUMERIC,\n    precision: constraints?.precision,\n    scale: constraints?.scale,\n    ...constraints,\n  }),\n\n  /**\n   * Create a DATE column (stored as TEXT)\n   */\n  date: (constraints?: ColumnConstraints): ColumnDefinition => ({\n    type: SQLiteTypes.DATE,\n    ...constraints,\n  }),\n\n  /**\n   * Create a DATETIME column (stored as TEXT)\n   */\n  datetime: (constraints?: ColumnConstraints): ColumnDefinition => ({\n    type: SQLiteTypes.DATETIME,\n    ...constraints,\n  }),\n\n  /**\n   * Create a TIMESTAMP column (stored as INTEGER by default)\n   */\n  timestamp: (\n    constraints?: ColumnConstraints & { asText?: boolean },\n  ): ColumnDefinition => ({\n    type: constraints?.asText ? SQLiteTypes.TEXT : SQLiteTypes.INTEGER,\n    ...constraints,\n  }),\n\n  /**\n   * Create a TIME column (stored as TEXT)\n   */\n  time: (constraints?: ColumnConstraints): ColumnDefinition => ({\n    type: SQLiteTypes.TIME,\n    ...constraints,\n  }),\n\n  /**\n   * Create a BOOLEAN column (stored as INTEGER)\n   */\n  boolean: (constraints?: ColumnConstraints): ColumnDefinition => ({\n    type: SQLiteTypes.BOOLEAN,\n    check: constraints?.check || \"{{COLUMN}} IN (0, 1)\", // Placeholder for column name\n    ...constraints,\n  }),\n\n  /**\n   * Create a JSON column (stored as TEXT)\n   */\n  json: (\n    constraints?: ColumnConstraints & { validateJson?: boolean },\n  ): ColumnDefinition => ({\n    type: SQLiteTypes.JSON,\n    check: constraints?.validateJson\n      ? \"JSON_VALID({{COLUMN}})\"\n      : constraints?.check,\n    ...constraints,\n  }),\n\n  /**\n   * Create a VARCHAR column with specified length\n   */\n  varchar: (\n    length: number,\n    constraints?: ColumnConstraints,\n  ): ColumnDefinition => ({\n    type: SQLiteTypes.VARCHAR,\n    length,\n    ...constraints,\n  }),\n\n  /**\n   * Create a CHAR column with specified length\n   */\n  char: (\n    length: number,\n    constraints?: ColumnConstraints,\n  ): ColumnDefinition => ({\n    type: SQLiteTypes.CHAR,\n    length,\n    ...constraints,\n  }),\n\n  /**\n   * Create an auto-incrementing primary key column\n   */\n  id: (\n    constraints?: Omit<\n      ColumnConstraints,\n      \"primaryKey\" | \"autoincrement\" | \"notNull\"\n    >,\n  ): ColumnDefinition => ({\n    type: SQLiteTypes.INTEGER,\n    primaryKey: true,\n    autoincrement: true,\n    notNull: true,\n    ...constraints,\n  }),\n\n  /**\n   * Create a UUID column (stored as TEXT)\n   */\n  uuid: (\n    constraints?: ColumnConstraints & { generateDefault?: boolean },\n  ): ColumnDefinition => ({\n    type: SQLiteTypes.TEXT,\n    length: 36,\n    default: constraints?.generateDefault\n      ? defaultExpr(\n          \"lower(hex(randomblob(4))) || '-' || lower(hex(randomblob(2))) || '-4' || substr(lower(hex(randomblob(2))),2) || '-' || substr('89ab',abs(random()) % 4 + 1, 1) || substr(lower(hex(randomblob(2))),2) || '-' || lower(hex(randomblob(6)))\",\n        )\n      : constraints?.default,\n    ...constraints,\n  }),\n\n  /**\n   * Create a created_at timestamp column\n   */\n  createdAt: (\n    constraints?: Omit<ColumnConstraints, \"default\" | \"notNull\"> & {\n      asText?: boolean;\n    },\n  ): ColumnDefinition => ({\n    type: constraints?.asText ? SQLiteTypes.DATETIME : SQLiteTypes.INTEGER,\n    notNull: true,\n    default: constraints?.asText\n      ? defaultExpr(\"datetime('now')\")\n      : defaultExpr(\"strftime('%s', 'now')\"),\n    ...constraints,\n  }),\n\n  /**\n   * Create an updated_at timestamp column\n   */\n  updatedAt: (\n    constraints?: Omit<ColumnConstraints, \"default\"> & { asText?: boolean },\n  ): ColumnDefinition => ({\n    type: constraints?.asText ? SQLiteTypes.DATETIME : SQLiteTypes.INTEGER,\n    default: constraints?.asText\n      ? defaultExpr(\"datetime('now')\")\n      : defaultExpr(\"strftime('%s', 'now')\"),\n    ...constraints,\n  }),\n\n  /**\n   * Create a foreign key reference column\n   */\n  foreignKey: (\n    refTable: string,\n    refColumn = \"id\",\n    constraints?: ColumnConstraints & {\n      onDelete?: ForeignKeyAction;\n      onUpdate?: ForeignKeyAction;\n      type?: SQLiteType;\n    },\n  ): ColumnDefinition => ({\n    type: constraints?.type || SQLiteTypes.INTEGER,\n    references: {\n      table: refTable,\n      column: refColumn,\n      onDelete: constraints?.onDelete,\n      onUpdate: constraints?.onUpdate,\n    },\n    ...constraints,\n  }),\n\n  /**\n   * Create an enum column (with CHECK constraint)\n   */\n  enum: (\n    values: string[],\n    constraints?: ColumnConstraints,\n  ): ColumnDefinition => ({\n    type: SQLiteTypes.TEXT,\n    check: `{{COLUMN}} IN (${values.map((v) => `'${v}'`).join(\", \")})`,\n    ...constraints,\n  }),\n};\n\n/**\n * SQL function helpers for use in defaults and expressions\n */\nexport const sql = {\n  // Current date/time\n  now: () => defaultExpr(\"datetime('now')\"),\n  currentTime: () => defaultExpr(\"time('now')\"),\n  currentDate: () => defaultExpr(\"date('now')\"),\n  currentTimestamp: () => defaultExpr(\"datetime('now')\"),\n  unixTimestamp: () => defaultExpr(\"strftime('%s', 'now')\"),\n\n  // Functions\n  ...SQLiteFunctions,\n\n  // Raw expression\n  raw: (expression: string) => defaultExpr(expression),\n\n  // Literals\n  null: () => null,\n  true: () => 1,\n  false: () => 0,\n};\n\n/**\n * Enhanced createTable method signature\n */\nexport type CreateTableColumns = string | Record<string, string> | TableSchema;\n\n/**\n * Query builder state interface\n */\nexport interface QueryBuilderState<T extends Record<string, unknown>> {\n  db: Database; // Database instance from bun:sqlite\n  tableName: string;\n  whereConditions: string[];\n  whereParams: SQLQueryBindings[];\n  regexConditions: Array<{\n    column: keyof T;\n    regex: RegExp;\n  }>;\n  jsonColumns?: Array<keyof T>;\n}\n\n/**\n * Column names type for SELECT operations\n */\nexport type ColumnNames<T> = [\"*\"] | Array<keyof T>;\n\n/**\n * Order direction for ORDER BY clauses\n */\nexport type OrderDirection = \"ASC\" | \"DESC\";\n\n/**\n * WHERE condition object type\n */\nexport type WhereCondition<T> = Partial<T>;\n\n/**\n * Regex condition object type\n */\nexport type RegexCondition<T> = Partial<Record<keyof T, RegExp | string>>;\n\n/**\n * Insert result interface\n */\nexport interface InsertResult {\n  insertId: number;\n  changes: number;\n}\n\n/**\n * Update result interface\n */\nexport interface UpdateResult {\n  changes: number;\n}\n\n/**\n * Delete result interface\n */\nexport interface DeleteResult {\n  changes: number;\n}\n\n/**\n * Insert options interface\n */\nexport interface InsertOptions {\n  orIgnore?: boolean;\n  orReplace?: boolean;\n  orAbort?: boolean;\n  orFail?: boolean;\n  orRollback?: boolean;\n}\n\n/**\n * JSON column configuration\n */\nexport interface JsonColumnConfig<T> {\n  jsonColumns?: Array<keyof T>;\n}\n\n/**\n * Generic database row type\n */\nexport type DatabaseRow = Record<string, unknown>;\n\n/**\n * SQL parameter type\n */\nexport type SqlParameter = SQLQueryBindings;\n\n/**\n * Database row with unknown structure\n */\nexport type DatabaseRowData = Record<string, SQLQueryBindings>;\n"
  ],
  "mappings": "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",
  "debugId": "7CE759C0B927428264756E2164756E21",
  "names": []
}