@event-driven-io/pongo 0.0.1

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.cjs ADDED
@@ -0,0 +1,2 @@
1
+ "use strict";Object.defineProperty(exports, "__esModule", {value: true}); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _nullishCoalesce(lhs, rhsFn) { if (lhs != null) { return lhs; } else { return rhsFn(); } } async function _asyncNullishCoalesce(lhs, rhsFn) { if (lhs != null) { return lhs; } else { return await rhsFn(); } } async function _asyncOptionalChain(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = await fn(value); } else if (op === 'call' || op === 'optionalCall') { value = await fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; }var _pg = require('pg'); var _pg2 = _interopRequireDefault(_pg);var _uuid = require('uuid');var _pgformat = require('pg-format'); var _pgformat2 = _interopRequireDefault(_pgformat);var c=async(t,e,...o)=>{let n=await t.connect();try{let r=_pgformat2.default.call(void 0, e,...o);return await n.query(r)}finally{n.release()}};var l=t=>Object.entries(t).map(([o,n])=>typeof n=="object"&&!Array.isArray(n)?x(o,n):m(o,n)).join(" AND "),m=(t,e)=>d(e)?_pgformat2.default.call(void 0, "(data->>'%I')::text = %L::text",t,e):f(e)?_pgformat2.default.call(void 0, "(data->>'%I')::timestamp = %L::timestamp",t,e):P(e)?_pgformat2.default.call(void 0, "(data->>'%I')::numeric = %L",t,e):_pgformat2.default.call(void 0, "(data->>'%I') = %L",t,e),x=(t,e)=>Object.entries(e).map(([n,r])=>{switch(n){case"$eq":return m(t,r);case"$gt":return u(t,r,">");case"$gte":return u(t,r,">=");case"$lt":return u(t,r,"<");case"$lte":return u(t,r,"<=");case"$ne":return m(t,r).replace("=","!=");case"$in":return _pgformat2.default.call(void 0, "(data->>'%I') IN (%s)",t,r.map(i=>C(t,i)).join(", "));case"$nin":return _pgformat2.default.call(void 0, "(data->>'%I') NOT IN (%s)",t,r.map(i=>C(t,i)).join(", "));default:throw new Error(`Unsupported operator: ${n}`)}}).join(" AND "),u=(t,e,o)=>d(e)?_pgformat2.default.call(void 0, `(data->>'%I')::text ${o} %L::text`,t,e):f(e)?_pgformat2.default.call(void 0, `(data->>'%I')::timestamp ${o} %L::timestamp`,t,e):P(e)?_pgformat2.default.call(void 0, `(data->>'%I')::numeric ${o} %s`,t,e):_pgformat2.default.call(void 0, `(data->>'%I') ${o} %L`,t,e),C=(t,e)=>d(e)?_pgformat2.default.call(void 0, "%L::text",e):f(e)?_pgformat2.default.call(void 0, "%L::timestamp",e):P(e)?_pgformat2.default.call(void 0, "%L",e):_pgformat2.default.call(void 0, "%L",e),d=t=>typeof t=="string"&&/^[0-9a-fA-F-]{36}$/.test(t),f=t=>typeof t=="string"&&!isNaN(Date.parse(t)),P=t=>typeof t=="number";var a=new Map,T= exports.getPool =t=>{let e=typeof t=="string"?t:t.connectionString,o=typeof t=="string"?{connectionString:e}:t;return _nullishCoalesce(a.get(e), () => (a.set(e,new _pg2.default.Pool(o)).get(e)))},L= exports.endPool =async t=>{let e=a.get(t);e&&(await e.end(),a.delete(t))},A= exports.endAllPools =()=>Promise.all([...a.keys()].map(t=>L(t)));var I=t=>{let e="data";if("$set"in t){let o=t.$set;e=_pgformat2.default.call(void 0, "jsonb_set(%s, %L, data || %L)",e,"{}",JSON.stringify(o))}if("$unset"in t){let o=Object.keys(t.$unset);e=_pgformat2.default.call(void 0, "%s - %L",e,o.join(", "))}if("$inc"in t){let o=t.$inc;for(let[n,r]of Object.entries(o))e=_pgformat2.default.call(void 0, "jsonb_set(%s, '{%s}', to_jsonb((data->>'%s')::numeric + %L))",e,n,n,r)}if("$push"in t){let o=t.$push;for(let[n,r]of Object.entries(o))e=_pgformat2.default.call(void 0, "jsonb_set(%s, '{%s}', (COALESCE(data->'%s', '[]'::jsonb) || to_jsonb(%L)))",e,n,n,r)}return e};var R=(t,e)=>{let o=T({connectionString:t,database:e});return{connect:()=>Promise.resolve(),close:()=>Promise.resolve(),collection:n=>F(n,o)}},F= exports.postgresCollection =(t,e)=>{let o=async()=>{await c(e,"CREATE TABLE IF NOT EXISTS %I (_id UUID PRIMARY KEY, data JSONB)",t)};return{createCollection:o,insertOne:async n=>{await o();let r=_uuid.v4.call(void 0, );return(await c(e,"INSERT INTO %I (_id, data) VALUES (%L, %L)",t,r,JSON.stringify({...n,_id:r}))).rowCount?{insertedId:r,acknowledged:!0}:{insertedId:null,acknowledged:!1}},updateOne:async(n,r)=>{let i=l(n),g=I(r),w=await c(e,"UPDATE %I SET data = %s WHERE %s",t,g,i);return w.rowCount?{acknowledged:!0,modifiedCount:w.rowCount}:{acknowledged:!1,modifiedCount:0}},deleteOne:async n=>{let r=l(n),i=await c(e,"DELETE FROM %I WHERE %s",t,r);return i.rowCount?{acknowledged:!0,deletedCount:i.rowCount}:{acknowledged:!1,deletedCount:0}},findOne:async n=>{let r=l(n);returnawait _asyncNullishCoalesce(await _asyncOptionalChain([(await c(e,"SELECT data FROM %I WHERE %s LIMIT 1",t,r)), 'access', async _ => _.rows, 'access', async _2 => _2[0], 'optionalAccess', async _3 => _3.data]), async () => (null))},find:async n=>{let r=l(n);return(await c(e,"SELECT data FROM %I WHERE %s",t,r)).rows.map(g=>g.data)}}};var y=(t,e)=>R(t,e);var v=t=>{let e=y(t),o={connect:async()=>(await e.connect(),o),close:()=>e.close(),db:n=>n?y(t,n):e};return o};exports.endAllPools = A; exports.endPool = L; exports.getDbClient = y; exports.getPool = T; exports.pongoClient = v; exports.postgresClient = R; exports.postgresCollection = F;
2
+ //# sourceMappingURL=index.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/postgres/client.ts","../src/postgres/execute/index.ts","../src/postgres/filter/index.ts","../src/postgres/pool.ts","../src/postgres/update/index.ts","../src/main/dbClient.ts","../src/main/client.ts"],"names":["uuid","format","sql","pool","sqlText","params","client","query","constructFilterQuery","filter","key","value","constructComplexFilterQuery","constructSimpleFilterQuery","isUUID","isDate","isNumber","operator","val","constructComparisonFilterQuery","v","constructSimpleFilterQueryValue","pg","pools","getPool","connectionStringOrOptions","connectionString","poolOptions","endPool","endAllPools","constructUpdateQuery","update","updateQuery","setUpdate","unsetUpdate","incUpdate","pushUpdate","postgresClient","database","name","postgresCollection","collectionName","createCollection","document","id","filterQuery","result","row","getDbClient","pongoClient","dbClient","dbName"],"mappings":"AAAA,MAAe,KACf,OAAS,MAAMA,MAAY,OCA3B,OAAOC,MAAY,YAEZ,IAAMC,EAAM,MACjBC,EACAC,KACGC,IACiC,CACpC,IAAMC,EAAS,MAAMH,EAAK,QAAQ,EAClC,GAAI,CACF,IAAMI,EAAQN,EAAOG,EAAS,GAAGC,CAAM,EACvC,OAAO,MAAMC,EAAO,MAAcC,CAAK,CACzC,QAAE,CACAD,EAAO,QAAQ,CACjB,CACF,ECdA,OAAOL,MAAY,YAGZ,IAAMO,EAA2BC,GACtB,OAAO,QAAQA,CAAM,EAAE,IAAI,CAAC,CAACC,EAAKC,CAAK,IACjD,OAAOA,GAAU,UAAY,CAAC,MAAM,QAAQA,CAAK,EAC5CC,EAA4BF,EAAKC,CAAgC,EAEjEE,EAA2BH,EAAKC,CAAK,CAE/C,EACc,KAAK,OAAO,EAGvBE,EAA6B,CAACH,EAAaC,IAC3CG,EAAOH,CAAK,EACPV,EAAO,iCAAkCS,EAAKC,CAAK,EACjDI,EAAOJ,CAAK,EACdV,EAAO,2CAA4CS,EAAKC,CAAK,EAC3DK,EAASL,CAAK,EAChBV,EAAO,8BAA+BS,EAAKC,CAAK,EAEhDV,EAAO,qBAAsBS,EAAKC,CAAK,EAIrCC,EAA8B,CACzCF,EACAC,IAEmB,OAAO,QAAQA,CAAK,EAAE,IAAI,CAAC,CAACM,EAAUC,CAAG,IAAM,CAChE,OAAQD,EAAU,CAChB,IAAK,MACH,OAAOJ,EAA2BH,EAAKQ,CAAG,EAC5C,IAAK,MACH,OAAOC,EAA+BT,EAAKQ,EAAK,GAAG,EACrD,IAAK,OACH,OAAOC,EAA+BT,EAAKQ,EAAK,IAAI,EACtD,IAAK,MACH,OAAOC,EAA+BT,EAAKQ,EAAK,GAAG,EACrD,IAAK,OACH,OAAOC,EAA+BT,EAAKQ,EAAK,IAAI,EACtD,IAAK,MACH,OAAOL,EAA2BH,EAAKQ,CAAG,EAAE,QAAQ,IAAK,IAAI,EAC/D,IAAK,MACH,OAAOjB,EACL,wBACAS,EACCQ,EACE,IAAKE,GAAMC,EAAgCX,EAAKU,CAAC,CAAC,EAClD,KAAK,IAAI,CACd,EACF,IAAK,OACH,OAAOnB,EACL,4BACAS,EACCQ,EACE,IAAKE,GAAMC,EAAgCX,EAAKU,CAAC,CAAC,EAClD,KAAK,IAAI,CACd,EACF,QACE,MAAM,IAAI,MAAM,yBAAyBH,CAAQ,EAAE,CACvD,CACF,CAAC,EACiB,KAAK,OAAO,EAG1BE,EAAiC,CACrCT,EACAC,EACAM,IAEIH,EAAOH,CAAK,EACPV,EAAO,uBAAuBgB,CAAQ,YAAaP,EAAKC,CAAK,EAC3DI,EAAOJ,CAAK,EACdV,EACL,4BAA4BgB,CAAQ,iBACpCP,EACAC,CACF,EACSK,EAASL,CAAK,EAChBV,EAAO,0BAA0BgB,CAAQ,MAAOP,EAAKC,CAAK,EAE1DV,EAAO,iBAAiBgB,CAAQ,MAAOP,EAAKC,CAAK,EAItDU,EAAkC,CACtCX,EACAC,IAEIG,EAAOH,CAAK,EACPV,EAAO,WAAYU,CAAK,EACtBI,EAAOJ,CAAK,EACdV,EAAO,gBAAiBU,CAAK,EAC3BK,EAASL,CAAK,EAChBV,EAAO,KAAMU,CAAK,EAElBV,EAAO,KAAMU,CAAK,EAIvBG,EAAUH,GACP,OAAOA,GAAU,UAAY,qBAAqB,KAAKA,CAAK,EAG/DI,EAAUJ,GACP,OAAOA,GAAU,UAAY,CAAC,MAAM,KAAK,MAAMA,CAAK,CAAC,EAGxDK,EAAYL,GACT,OAAOA,GAAU,SChH1B,OAAOW,MAAQ,KAEf,IAAMC,EAA8B,IAAI,IAE3BC,EACXC,GACY,CACZ,IAAMC,EACJ,OAAOD,GAA8B,SACjCA,EACAA,EAA0B,iBAE1BE,EACJ,OAAOF,GAA8B,SACjC,CAAE,iBAAAC,CAAiB,EACnBD,EAEN,OACEF,EAAM,IAAIG,CAAgB,GAC1BH,EAAM,IAAIG,EAAkB,IAAIJ,EAAG,KAAKK,CAAW,CAAC,EAAE,IAAID,CAAgB,CAE9E,EAEaE,EAAU,MAAOF,GAA4C,CACxE,IAAMvB,EAAOoB,EAAM,IAAIG,CAAgB,EACnCvB,IACF,MAAMA,EAAK,IAAI,EACfoB,EAAM,OAAOG,CAAgB,EAEjC,EAEaG,EAAc,IACzB,QAAQ,IACN,CAAC,GAAGN,EAAM,KAAK,CAAC,EAAE,IAAKG,GAAqBE,EAAQF,CAAgB,CAAC,CACvE,EClCF,OAAOzB,MAAY,YAGZ,IAAM6B,EAA2BC,GAAmC,CACzE,IAAIC,EAAc,OAElB,GAAI,SAAUD,EAAQ,CACpB,IAAME,EAAYF,EAAO,KACzBC,EAAc/B,EACZ,gCACA+B,EACA,KACA,KAAK,UAAUC,CAAS,CAC1B,CACF,CAEA,GAAI,WAAYF,EAAQ,CACtB,IAAMG,EAAc,OAAO,KAAKH,EAAO,MAAO,EAC9CC,EAAc/B,EAAO,UAAW+B,EAAaE,EAAY,KAAK,IAAI,CAAC,CACrE,CAEA,GAAI,SAAUH,EAAQ,CACpB,IAAMI,EAAYJ,EAAO,KACzB,OAAW,CAACrB,EAAKC,CAAK,IAAK,OAAO,QAAQwB,CAAS,EACjDH,EAAc/B,EACZ,+DACA+B,EACAtB,EACAA,EACAC,CACF,CAEJ,CAEA,GAAI,UAAWoB,EAAQ,CACrB,IAAMK,EAAaL,EAAO,MAC1B,OAAW,CAACrB,EAAKC,CAAK,IAAK,OAAO,QAAQyB,CAAU,EAClDJ,EAAc/B,EACZ,6EACA+B,EACAtB,EACAA,EACAC,CACF,CAEJ,CAEA,OAAOqB,CACT,EJhCO,IAAMK,EAAiB,CAC5BX,EACAY,IACa,CACb,IAAMnC,EAAOqB,EAAQ,CAAE,iBAAAE,EAAkB,SAAAY,CAAS,CAAC,EAEnD,MAAO,CACL,QAAS,IAAM,QAAQ,QAAQ,EAC/B,MAAO,IAAM,QAAQ,QAAQ,EAC7B,WAAgBC,GAAiBC,EAAsBD,EAAMpC,CAAI,CACnE,CACF,EAEaqC,EAAqB,CAChCC,EACAtC,IACuB,CACvB,IAAMuC,EAAmB,SAA2B,CAClD,MAAMxC,EACJC,EACA,mEACAsC,CACF,CACF,EAEA,MAAO,CACL,iBAAAC,EACA,UAAW,MAAOC,GAA+C,CAC/D,MAAMD,EAAiB,EAEvB,IAAME,EAAK5C,EAAK,EAUhB,OARe,MAAME,EACnBC,EACA,6CACAsC,EACAG,EACA,KAAK,UAAU,CAAE,GAAGD,EAAU,IAAKC,CAAG,CAAC,CACzC,GAEc,SACV,CAAE,WAAYA,EAAI,aAAc,EAAK,EACrC,CAAE,WAAY,KAAM,aAAc,EAAM,CAC9C,EACA,UAAW,MACTnC,EACAsB,IAC+B,CAC/B,IAAMc,EAAcrC,EAAqBC,CAAM,EACzCuB,EAAcF,EAAqBC,CAAM,EAEzCe,EAAS,MAAM5C,EACnBC,EACA,mCACAsC,EACAT,EACAa,CACF,EACA,OAAOC,EAAO,SACV,CAAE,aAAc,GAAM,cAAeA,EAAO,QAAS,EACrD,CAAE,aAAc,GAAO,cAAe,CAAE,CAC9C,EACA,UAAW,MAAOrC,GAAuD,CACvE,IAAMoC,EAAcrC,EAAqBC,CAAM,EACzCqC,EAAS,MAAM5C,EACnBC,EACA,0BACAsC,EACAI,CACF,EACA,OAAOC,EAAO,SACV,CAAE,aAAc,GAAM,aAAcA,EAAO,QAAS,EACpD,CAAE,aAAc,GAAO,aAAc,CAAE,CAC7C,EACA,QAAS,MAAOrC,GAA8C,CAC5D,IAAMoC,EAAcrC,EAAqBC,CAAM,EAO/C,OANe,MAAMP,EACnBC,EACA,uCACAsC,EACAI,CACF,GACe,KAAK,CAAC,GAAG,MAAQ,IAClC,EACA,KAAM,MAAOpC,GAAyC,CACpD,IAAMoC,EAAcrC,EAAqBC,CAAM,EAQ/C,OAPe,MAAMP,EACnBC,EACA,+BACAsC,EACAI,CACF,GAEc,KAAK,IAAKE,GAAQA,EAAI,IAAS,CAC/C,CACF,CACF,EKvGO,IAAMC,EAAc,CACzBtB,EACAY,IAGOD,EAAeX,EAAkBY,CAAQ,ECX3C,IAAMW,EAAevB,GAA0C,CACpE,IAAMwB,EAAWF,EAAYtB,CAAgB,EAEvCuB,EAA2B,CAC/B,QAAS,UACP,MAAMC,EAAS,QAAQ,EAChBD,GAET,MAAO,IAAMC,EAAS,MAAM,EAC5B,GAAKC,GACHA,EAASH,EAAYtB,EAAkByB,CAAM,EAAID,CACrD,EAEA,OAAOD,CACT","sourcesContent":["import pg from 'pg';\nimport { v4 as uuid } from 'uuid';\nimport {\n type DbClient,\n type PongoCollection,\n type PongoDeleteResult,\n type PongoFilter,\n type PongoInsertOneResult,\n type PongoUpdate,\n type PongoUpdateResult,\n} from '../main';\nimport { sql } from './execute';\nimport { constructFilterQuery } from './filter';\nimport { getPool } from './pool';\nimport { constructUpdateQuery } from './update';\n\nexport const postgresClient = (\n connectionString: string,\n database?: string,\n): DbClient => {\n const pool = getPool({ connectionString, database });\n\n return {\n connect: () => Promise.resolve(),\n close: () => Promise.resolve(),\n collection: <T>(name: string) => postgresCollection<T>(name, pool),\n };\n};\n\nexport const postgresCollection = <T>(\n collectionName: string,\n pool: pg.Pool,\n): PongoCollection<T> => {\n const createCollection = async (): Promise<void> => {\n await sql(\n pool,\n 'CREATE TABLE IF NOT EXISTS %I (_id UUID PRIMARY KEY, data JSONB)',\n collectionName,\n );\n };\n\n return {\n createCollection,\n insertOne: async (document: T): Promise<PongoInsertOneResult> => {\n await createCollection();\n\n const id = uuid();\n\n const result = await sql(\n pool,\n 'INSERT INTO %I (_id, data) VALUES (%L, %L)',\n collectionName,\n id,\n JSON.stringify({ ...document, _id: id }),\n );\n\n return result.rowCount\n ? { insertedId: id, acknowledged: true }\n : { insertedId: null, acknowledged: false };\n },\n updateOne: async (\n filter: PongoFilter<T>,\n update: PongoUpdate<T>,\n ): Promise<PongoUpdateResult> => {\n const filterQuery = constructFilterQuery(filter);\n const updateQuery = constructUpdateQuery(update);\n\n const result = await sql(\n pool,\n 'UPDATE %I SET data = %s WHERE %s',\n collectionName,\n updateQuery,\n filterQuery,\n );\n return result.rowCount\n ? { acknowledged: true, modifiedCount: result.rowCount }\n : { acknowledged: false, modifiedCount: 0 };\n },\n deleteOne: async (filter: PongoFilter<T>): Promise<PongoDeleteResult> => {\n const filterQuery = constructFilterQuery(filter);\n const result = await sql(\n pool,\n 'DELETE FROM %I WHERE %s',\n collectionName,\n filterQuery,\n );\n return result.rowCount\n ? { acknowledged: true, deletedCount: result.rowCount }\n : { acknowledged: false, deletedCount: 0 };\n },\n findOne: async (filter: PongoFilter<T>): Promise<T | null> => {\n const filterQuery = constructFilterQuery(filter);\n const result = await sql(\n pool,\n 'SELECT data FROM %I WHERE %s LIMIT 1',\n collectionName,\n filterQuery,\n );\n return (result.rows[0]?.data ?? null) as T | null;\n },\n find: async (filter: PongoFilter<T>): Promise<T[]> => {\n const filterQuery = constructFilterQuery(filter);\n const result = await sql(\n pool,\n 'SELECT data FROM %I WHERE %s',\n collectionName,\n filterQuery,\n );\n\n return result.rows.map((row) => row.data as T);\n },\n };\n};\n","import type pg from 'pg';\nimport format from 'pg-format';\n\nexport const sql = async <Result extends pg.QueryResultRow = pg.QueryResultRow>(\n pool: pg.Pool,\n sqlText: string,\n ...params: unknown[]\n): Promise<pg.QueryResult<Result>> => {\n const client = await pool.connect();\n try {\n const query = format(sqlText, ...params);\n return await client.query<Result>(query);\n } finally {\n client.release();\n }\n};\n\nexport const execute = async <Result = void>(\n pool: pg.Pool,\n handle: (client: pg.PoolClient) => Promise<Result>,\n) => {\n const client = await pool.connect();\n try {\n return await handle(client);\n } finally {\n client.release();\n }\n};\n","// src/utils.ts\nimport format from 'pg-format';\nimport type { PongoFilter } from '../../main';\n\nexport const constructFilterQuery = <T>(filter: PongoFilter<T>): string => {\n const filters = Object.entries(filter).map(([key, value]) => {\n if (typeof value === 'object' && !Array.isArray(value)) {\n return constructComplexFilterQuery(key, value as Record<string, unknown>);\n } else {\n return constructSimpleFilterQuery(key, value);\n }\n });\n return filters.join(' AND ');\n};\n\nconst constructSimpleFilterQuery = (key: string, value: unknown): string => {\n if (isUUID(value)) {\n return format(`(data->>'%I')::text = %L::text`, key, value);\n } else if (isDate(value)) {\n return format(`(data->>'%I')::timestamp = %L::timestamp`, key, value);\n } else if (isNumber(value)) {\n return format(`(data->>'%I')::numeric = %L`, key, value);\n } else {\n return format(`(data->>'%I') = %L`, key, value);\n }\n};\n\nexport const constructComplexFilterQuery = (\n key: string,\n value: Record<string, unknown>,\n): string => {\n const subFilters = Object.entries(value).map(([operator, val]) => {\n switch (operator) {\n case '$eq':\n return constructSimpleFilterQuery(key, val);\n case '$gt':\n return constructComparisonFilterQuery(key, val, '>');\n case '$gte':\n return constructComparisonFilterQuery(key, val, '>=');\n case '$lt':\n return constructComparisonFilterQuery(key, val, '<');\n case '$lte':\n return constructComparisonFilterQuery(key, val, '<=');\n case '$ne':\n return constructSimpleFilterQuery(key, val).replace('=', '!=');\n case '$in':\n return format(\n `(data->>'%I') IN (%s)`,\n key,\n (val as unknown[])\n .map((v) => constructSimpleFilterQueryValue(key, v))\n .join(', '),\n );\n case '$nin':\n return format(\n `(data->>'%I') NOT IN (%s)`,\n key,\n (val as unknown[])\n .map((v) => constructSimpleFilterQueryValue(key, v))\n .join(', '),\n );\n default:\n throw new Error(`Unsupported operator: ${operator}`);\n }\n });\n return subFilters.join(' AND ');\n};\n\nconst constructComparisonFilterQuery = (\n key: string,\n value: unknown,\n operator: string,\n): string => {\n if (isUUID(value)) {\n return format(`(data->>'%I')::text ${operator} %L::text`, key, value);\n } else if (isDate(value)) {\n return format(\n `(data->>'%I')::timestamp ${operator} %L::timestamp`,\n key,\n value,\n );\n } else if (isNumber(value)) {\n return format(`(data->>'%I')::numeric ${operator} %s`, key, value);\n } else {\n return format(`(data->>'%I') ${operator} %L`, key, value);\n }\n};\n\nconst constructSimpleFilterQueryValue = (\n key: string,\n value: unknown,\n): string => {\n if (isUUID(value)) {\n return format('%L::text', value);\n } else if (isDate(value)) {\n return format('%L::timestamp', value);\n } else if (isNumber(value)) {\n return format('%L', value);\n } else {\n return format('%L', value);\n }\n};\n\nconst isUUID = (value: unknown): boolean => {\n return typeof value === 'string' && /^[0-9a-fA-F-]{36}$/.test(value);\n};\n\nconst isDate = (value: unknown): boolean => {\n return typeof value === 'string' && !isNaN(Date.parse(value));\n};\n\nconst isNumber = (value: unknown): boolean => {\n return typeof value === 'number';\n};\n","import pg from 'pg';\n\nconst pools: Map<string, pg.Pool> = new Map();\n\nexport const getPool = (\n connectionStringOrOptions: string | pg.PoolConfig,\n): pg.Pool => {\n const connectionString =\n typeof connectionStringOrOptions === 'string'\n ? connectionStringOrOptions\n : connectionStringOrOptions.connectionString!;\n\n const poolOptions =\n typeof connectionStringOrOptions === 'string'\n ? { connectionString }\n : connectionStringOrOptions;\n\n return (\n pools.get(connectionString) ??\n pools.set(connectionString, new pg.Pool(poolOptions)).get(connectionString)!\n );\n};\n\nexport const endPool = async (connectionString: string): Promise<void> => {\n const pool = pools.get(connectionString);\n if (pool) {\n await pool.end();\n pools.delete(connectionString);\n }\n};\n\nexport const endAllPools = () =>\n Promise.all(\n [...pools.keys()].map((connectionString) => endPool(connectionString)),\n );\n","import format from 'pg-format';\nimport type { PongoUpdate } from '../../main';\n\nexport const constructUpdateQuery = <T>(update: PongoUpdate<T>): string => {\n let updateQuery = 'data';\n\n if ('$set' in update) {\n const setUpdate = update.$set!;\n updateQuery = format(\n 'jsonb_set(%s, %L, data || %L)',\n updateQuery,\n '{}',\n JSON.stringify(setUpdate),\n );\n }\n\n if ('$unset' in update) {\n const unsetUpdate = Object.keys(update.$unset!);\n updateQuery = format('%s - %L', updateQuery, unsetUpdate.join(', '));\n }\n\n if ('$inc' in update) {\n const incUpdate = update.$inc!;\n for (const [key, value] of Object.entries(incUpdate)) {\n updateQuery = format(\n \"jsonb_set(%s, '{%s}', to_jsonb((data->>'%s')::numeric + %L))\",\n updateQuery,\n key,\n key,\n value,\n );\n }\n }\n\n if ('$push' in update) {\n const pushUpdate = update.$push!;\n for (const [key, value] of Object.entries(pushUpdate)) {\n updateQuery = format(\n \"jsonb_set(%s, '{%s}', (COALESCE(data->'%s', '[]'::jsonb) || to_jsonb(%L)))\",\n updateQuery,\n key,\n key,\n value,\n );\n }\n }\n\n return updateQuery;\n};\n","import { postgresClient } from '../postgres';\nimport type { PongoCollection } from './typing';\n\nexport interface DbClient {\n connect(): Promise<void>;\n close(): Promise<void>;\n collection: <T>(name: string) => PongoCollection<T>;\n}\n\nexport const getDbClient = (\n connectionString: string,\n database?: string,\n): DbClient => {\n // This is the place where in the future could come resolution of other database types\n return postgresClient(connectionString, database);\n};\n","import { getDbClient } from './dbClient';\nimport type { PongoClient, PongoDb } from './typing';\n\nexport const pongoClient = (connectionString: string): PongoClient => {\n const dbClient = getDbClient(connectionString);\n\n const pongoClient: PongoClient = {\n connect: async () => {\n await dbClient.connect();\n return pongoClient;\n },\n close: () => dbClient.close(),\n db: (dbName?: string): PongoDb =>\n dbName ? getDbClient(connectionString, dbName) : dbClient,\n };\n\n return pongoClient;\n};\n"]}
@@ -0,0 +1,73 @@
1
+ import pg from 'pg';
2
+
3
+ interface PongoClient {
4
+ connect(): Promise<this>;
5
+ close(): Promise<void>;
6
+ db(dbName?: string): PongoDb;
7
+ }
8
+ interface PongoDb {
9
+ collection<T>(name: string): PongoCollection<T>;
10
+ }
11
+ interface PongoCollection<T> {
12
+ createCollection(): Promise<void>;
13
+ insertOne(document: T): Promise<PongoInsertOneResult>;
14
+ updateOne(filter: PongoFilter<T>, update: PongoUpdate<T>): Promise<PongoUpdateResult>;
15
+ deleteOne(filter: PongoFilter<T>): Promise<PongoDeleteResult>;
16
+ findOne(filter: PongoFilter<T>): Promise<T | null>;
17
+ find(filter: PongoFilter<T>): Promise<T[]>;
18
+ }
19
+ type PongoFilter<T> = {
20
+ [P in keyof T]?: T[P] | PongoFilterOperator<T[P]>;
21
+ };
22
+ type PongoFilterOperator<T> = {
23
+ $eq?: T;
24
+ $gt?: T;
25
+ $gte?: T;
26
+ $lt?: T;
27
+ $lte?: T;
28
+ $ne?: T;
29
+ $in?: T[];
30
+ $nin?: T[];
31
+ };
32
+ type PongoUpdate<T> = {
33
+ $set?: Partial<T>;
34
+ $unset?: {
35
+ [P in keyof T]?: '';
36
+ };
37
+ $inc?: {
38
+ [P in keyof T]?: number;
39
+ };
40
+ $push?: {
41
+ [P in keyof T]?: T[P];
42
+ };
43
+ };
44
+ interface PongoInsertOneResult {
45
+ insertedId: string | null;
46
+ acknowledged: boolean;
47
+ }
48
+ interface PongoUpdateResult {
49
+ acknowledged: boolean;
50
+ modifiedCount: number;
51
+ }
52
+ interface PongoDeleteResult {
53
+ acknowledged: boolean;
54
+ deletedCount: number;
55
+ }
56
+
57
+ declare const pongoClient: (connectionString: string) => PongoClient;
58
+
59
+ interface DbClient {
60
+ connect(): Promise<void>;
61
+ close(): Promise<void>;
62
+ collection: <T>(name: string) => PongoCollection<T>;
63
+ }
64
+ declare const getDbClient: (connectionString: string, database?: string) => DbClient;
65
+
66
+ declare const postgresClient: (connectionString: string, database?: string) => DbClient;
67
+ declare const postgresCollection: <T>(collectionName: string, pool: pg.Pool) => PongoCollection<T>;
68
+
69
+ declare const getPool: (connectionStringOrOptions: string | pg.PoolConfig) => pg.Pool;
70
+ declare const endPool: (connectionString: string) => Promise<void>;
71
+ declare const endAllPools: () => Promise<void[]>;
72
+
73
+ export { type DbClient, type PongoClient, type PongoCollection, type PongoDb, type PongoDeleteResult, type PongoFilter, type PongoFilterOperator, type PongoInsertOneResult, type PongoUpdate, type PongoUpdateResult, endAllPools, endPool, getDbClient, getPool, pongoClient, postgresClient, postgresCollection };
@@ -0,0 +1,73 @@
1
+ import pg from 'pg';
2
+
3
+ interface PongoClient {
4
+ connect(): Promise<this>;
5
+ close(): Promise<void>;
6
+ db(dbName?: string): PongoDb;
7
+ }
8
+ interface PongoDb {
9
+ collection<T>(name: string): PongoCollection<T>;
10
+ }
11
+ interface PongoCollection<T> {
12
+ createCollection(): Promise<void>;
13
+ insertOne(document: T): Promise<PongoInsertOneResult>;
14
+ updateOne(filter: PongoFilter<T>, update: PongoUpdate<T>): Promise<PongoUpdateResult>;
15
+ deleteOne(filter: PongoFilter<T>): Promise<PongoDeleteResult>;
16
+ findOne(filter: PongoFilter<T>): Promise<T | null>;
17
+ find(filter: PongoFilter<T>): Promise<T[]>;
18
+ }
19
+ type PongoFilter<T> = {
20
+ [P in keyof T]?: T[P] | PongoFilterOperator<T[P]>;
21
+ };
22
+ type PongoFilterOperator<T> = {
23
+ $eq?: T;
24
+ $gt?: T;
25
+ $gte?: T;
26
+ $lt?: T;
27
+ $lte?: T;
28
+ $ne?: T;
29
+ $in?: T[];
30
+ $nin?: T[];
31
+ };
32
+ type PongoUpdate<T> = {
33
+ $set?: Partial<T>;
34
+ $unset?: {
35
+ [P in keyof T]?: '';
36
+ };
37
+ $inc?: {
38
+ [P in keyof T]?: number;
39
+ };
40
+ $push?: {
41
+ [P in keyof T]?: T[P];
42
+ };
43
+ };
44
+ interface PongoInsertOneResult {
45
+ insertedId: string | null;
46
+ acknowledged: boolean;
47
+ }
48
+ interface PongoUpdateResult {
49
+ acknowledged: boolean;
50
+ modifiedCount: number;
51
+ }
52
+ interface PongoDeleteResult {
53
+ acknowledged: boolean;
54
+ deletedCount: number;
55
+ }
56
+
57
+ declare const pongoClient: (connectionString: string) => PongoClient;
58
+
59
+ interface DbClient {
60
+ connect(): Promise<void>;
61
+ close(): Promise<void>;
62
+ collection: <T>(name: string) => PongoCollection<T>;
63
+ }
64
+ declare const getDbClient: (connectionString: string, database?: string) => DbClient;
65
+
66
+ declare const postgresClient: (connectionString: string, database?: string) => DbClient;
67
+ declare const postgresCollection: <T>(collectionName: string, pool: pg.Pool) => PongoCollection<T>;
68
+
69
+ declare const getPool: (connectionStringOrOptions: string | pg.PoolConfig) => pg.Pool;
70
+ declare const endPool: (connectionString: string) => Promise<void>;
71
+ declare const endAllPools: () => Promise<void[]>;
72
+
73
+ export { type DbClient, type PongoClient, type PongoCollection, type PongoDb, type PongoDeleteResult, type PongoFilter, type PongoFilterOperator, type PongoInsertOneResult, type PongoUpdate, type PongoUpdateResult, endAllPools, endPool, getDbClient, getPool, pongoClient, postgresClient, postgresCollection };
package/dist/index.js ADDED
@@ -0,0 +1,2 @@
1
+ import"pg";import{v4 as $}from"uuid";import b from"pg-format";var c=async(t,e,...o)=>{let n=await t.connect();try{let r=b(e,...o);return await n.query(r)}finally{n.release()}};import s from"pg-format";var l=t=>Object.entries(t).map(([o,n])=>typeof n=="object"&&!Array.isArray(n)?x(o,n):m(o,n)).join(" AND "),m=(t,e)=>d(e)?s("(data->>'%I')::text = %L::text",t,e):f(e)?s("(data->>'%I')::timestamp = %L::timestamp",t,e):P(e)?s("(data->>'%I')::numeric = %L",t,e):s("(data->>'%I') = %L",t,e),x=(t,e)=>Object.entries(e).map(([n,r])=>{switch(n){case"$eq":return m(t,r);case"$gt":return u(t,r,">");case"$gte":return u(t,r,">=");case"$lt":return u(t,r,"<");case"$lte":return u(t,r,"<=");case"$ne":return m(t,r).replace("=","!=");case"$in":return s("(data->>'%I') IN (%s)",t,r.map(i=>C(t,i)).join(", "));case"$nin":return s("(data->>'%I') NOT IN (%s)",t,r.map(i=>C(t,i)).join(", "));default:throw new Error(`Unsupported operator: ${n}`)}}).join(" AND "),u=(t,e,o)=>d(e)?s(`(data->>'%I')::text ${o} %L::text`,t,e):f(e)?s(`(data->>'%I')::timestamp ${o} %L::timestamp`,t,e):P(e)?s(`(data->>'%I')::numeric ${o} %s`,t,e):s(`(data->>'%I') ${o} %L`,t,e),C=(t,e)=>d(e)?s("%L::text",e):f(e)?s("%L::timestamp",e):P(e)?s("%L",e):s("%L",e),d=t=>typeof t=="string"&&/^[0-9a-fA-F-]{36}$/.test(t),f=t=>typeof t=="string"&&!isNaN(Date.parse(t)),P=t=>typeof t=="number";import E from"pg";var a=new Map,T=t=>{let e=typeof t=="string"?t:t.connectionString,o=typeof t=="string"?{connectionString:e}:t;return a.get(e)??a.set(e,new E.Pool(o)).get(e)},L=async t=>{let e=a.get(t);e&&(await e.end(),a.delete(t))},A=()=>Promise.all([...a.keys()].map(t=>L(t)));import p from"pg-format";var I=t=>{let e="data";if("$set"in t){let o=t.$set;e=p("jsonb_set(%s, %L, data || %L)",e,"{}",JSON.stringify(o))}if("$unset"in t){let o=Object.keys(t.$unset);e=p("%s - %L",e,o.join(", "))}if("$inc"in t){let o=t.$inc;for(let[n,r]of Object.entries(o))e=p("jsonb_set(%s, '{%s}', to_jsonb((data->>'%s')::numeric + %L))",e,n,n,r)}if("$push"in t){let o=t.$push;for(let[n,r]of Object.entries(o))e=p("jsonb_set(%s, '{%s}', (COALESCE(data->'%s', '[]'::jsonb) || to_jsonb(%L)))",e,n,n,r)}return e};var R=(t,e)=>{let o=T({connectionString:t,database:e});return{connect:()=>Promise.resolve(),close:()=>Promise.resolve(),collection:n=>F(n,o)}},F=(t,e)=>{let o=async()=>{await c(e,"CREATE TABLE IF NOT EXISTS %I (_id UUID PRIMARY KEY, data JSONB)",t)};return{createCollection:o,insertOne:async n=>{await o();let r=$();return(await c(e,"INSERT INTO %I (_id, data) VALUES (%L, %L)",t,r,JSON.stringify({...n,_id:r}))).rowCount?{insertedId:r,acknowledged:!0}:{insertedId:null,acknowledged:!1}},updateOne:async(n,r)=>{let i=l(n),g=I(r),w=await c(e,"UPDATE %I SET data = %s WHERE %s",t,g,i);return w.rowCount?{acknowledged:!0,modifiedCount:w.rowCount}:{acknowledged:!1,modifiedCount:0}},deleteOne:async n=>{let r=l(n),i=await c(e,"DELETE FROM %I WHERE %s",t,r);return i.rowCount?{acknowledged:!0,deletedCount:i.rowCount}:{acknowledged:!1,deletedCount:0}},findOne:async n=>{let r=l(n);return(await c(e,"SELECT data FROM %I WHERE %s LIMIT 1",t,r)).rows[0]?.data??null},find:async n=>{let r=l(n);return(await c(e,"SELECT data FROM %I WHERE %s",t,r)).rows.map(g=>g.data)}}};var y=(t,e)=>R(t,e);var v=t=>{let e=y(t),o={connect:async()=>(await e.connect(),o),close:()=>e.close(),db:n=>n?y(t,n):e};return o};export{A as endAllPools,L as endPool,y as getDbClient,T as getPool,v as pongoClient,R as postgresClient,F as postgresCollection};
2
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/postgres/client.ts","../src/postgres/execute/index.ts","../src/postgres/filter/index.ts","../src/postgres/pool.ts","../src/postgres/update/index.ts","../src/main/dbClient.ts","../src/main/client.ts"],"sourcesContent":["import pg from 'pg';\nimport { v4 as uuid } from 'uuid';\nimport {\n type DbClient,\n type PongoCollection,\n type PongoDeleteResult,\n type PongoFilter,\n type PongoInsertOneResult,\n type PongoUpdate,\n type PongoUpdateResult,\n} from '../main';\nimport { sql } from './execute';\nimport { constructFilterQuery } from './filter';\nimport { getPool } from './pool';\nimport { constructUpdateQuery } from './update';\n\nexport const postgresClient = (\n connectionString: string,\n database?: string,\n): DbClient => {\n const pool = getPool({ connectionString, database });\n\n return {\n connect: () => Promise.resolve(),\n close: () => Promise.resolve(),\n collection: <T>(name: string) => postgresCollection<T>(name, pool),\n };\n};\n\nexport const postgresCollection = <T>(\n collectionName: string,\n pool: pg.Pool,\n): PongoCollection<T> => {\n const createCollection = async (): Promise<void> => {\n await sql(\n pool,\n 'CREATE TABLE IF NOT EXISTS %I (_id UUID PRIMARY KEY, data JSONB)',\n collectionName,\n );\n };\n\n return {\n createCollection,\n insertOne: async (document: T): Promise<PongoInsertOneResult> => {\n await createCollection();\n\n const id = uuid();\n\n const result = await sql(\n pool,\n 'INSERT INTO %I (_id, data) VALUES (%L, %L)',\n collectionName,\n id,\n JSON.stringify({ ...document, _id: id }),\n );\n\n return result.rowCount\n ? { insertedId: id, acknowledged: true }\n : { insertedId: null, acknowledged: false };\n },\n updateOne: async (\n filter: PongoFilter<T>,\n update: PongoUpdate<T>,\n ): Promise<PongoUpdateResult> => {\n const filterQuery = constructFilterQuery(filter);\n const updateQuery = constructUpdateQuery(update);\n\n const result = await sql(\n pool,\n 'UPDATE %I SET data = %s WHERE %s',\n collectionName,\n updateQuery,\n filterQuery,\n );\n return result.rowCount\n ? { acknowledged: true, modifiedCount: result.rowCount }\n : { acknowledged: false, modifiedCount: 0 };\n },\n deleteOne: async (filter: PongoFilter<T>): Promise<PongoDeleteResult> => {\n const filterQuery = constructFilterQuery(filter);\n const result = await sql(\n pool,\n 'DELETE FROM %I WHERE %s',\n collectionName,\n filterQuery,\n );\n return result.rowCount\n ? { acknowledged: true, deletedCount: result.rowCount }\n : { acknowledged: false, deletedCount: 0 };\n },\n findOne: async (filter: PongoFilter<T>): Promise<T | null> => {\n const filterQuery = constructFilterQuery(filter);\n const result = await sql(\n pool,\n 'SELECT data FROM %I WHERE %s LIMIT 1',\n collectionName,\n filterQuery,\n );\n return (result.rows[0]?.data ?? null) as T | null;\n },\n find: async (filter: PongoFilter<T>): Promise<T[]> => {\n const filterQuery = constructFilterQuery(filter);\n const result = await sql(\n pool,\n 'SELECT data FROM %I WHERE %s',\n collectionName,\n filterQuery,\n );\n\n return result.rows.map((row) => row.data as T);\n },\n };\n};\n","import type pg from 'pg';\nimport format from 'pg-format';\n\nexport const sql = async <Result extends pg.QueryResultRow = pg.QueryResultRow>(\n pool: pg.Pool,\n sqlText: string,\n ...params: unknown[]\n): Promise<pg.QueryResult<Result>> => {\n const client = await pool.connect();\n try {\n const query = format(sqlText, ...params);\n return await client.query<Result>(query);\n } finally {\n client.release();\n }\n};\n\nexport const execute = async <Result = void>(\n pool: pg.Pool,\n handle: (client: pg.PoolClient) => Promise<Result>,\n) => {\n const client = await pool.connect();\n try {\n return await handle(client);\n } finally {\n client.release();\n }\n};\n","// src/utils.ts\nimport format from 'pg-format';\nimport type { PongoFilter } from '../../main';\n\nexport const constructFilterQuery = <T>(filter: PongoFilter<T>): string => {\n const filters = Object.entries(filter).map(([key, value]) => {\n if (typeof value === 'object' && !Array.isArray(value)) {\n return constructComplexFilterQuery(key, value as Record<string, unknown>);\n } else {\n return constructSimpleFilterQuery(key, value);\n }\n });\n return filters.join(' AND ');\n};\n\nconst constructSimpleFilterQuery = (key: string, value: unknown): string => {\n if (isUUID(value)) {\n return format(`(data->>'%I')::text = %L::text`, key, value);\n } else if (isDate(value)) {\n return format(`(data->>'%I')::timestamp = %L::timestamp`, key, value);\n } else if (isNumber(value)) {\n return format(`(data->>'%I')::numeric = %L`, key, value);\n } else {\n return format(`(data->>'%I') = %L`, key, value);\n }\n};\n\nexport const constructComplexFilterQuery = (\n key: string,\n value: Record<string, unknown>,\n): string => {\n const subFilters = Object.entries(value).map(([operator, val]) => {\n switch (operator) {\n case '$eq':\n return constructSimpleFilterQuery(key, val);\n case '$gt':\n return constructComparisonFilterQuery(key, val, '>');\n case '$gte':\n return constructComparisonFilterQuery(key, val, '>=');\n case '$lt':\n return constructComparisonFilterQuery(key, val, '<');\n case '$lte':\n return constructComparisonFilterQuery(key, val, '<=');\n case '$ne':\n return constructSimpleFilterQuery(key, val).replace('=', '!=');\n case '$in':\n return format(\n `(data->>'%I') IN (%s)`,\n key,\n (val as unknown[])\n .map((v) => constructSimpleFilterQueryValue(key, v))\n .join(', '),\n );\n case '$nin':\n return format(\n `(data->>'%I') NOT IN (%s)`,\n key,\n (val as unknown[])\n .map((v) => constructSimpleFilterQueryValue(key, v))\n .join(', '),\n );\n default:\n throw new Error(`Unsupported operator: ${operator}`);\n }\n });\n return subFilters.join(' AND ');\n};\n\nconst constructComparisonFilterQuery = (\n key: string,\n value: unknown,\n operator: string,\n): string => {\n if (isUUID(value)) {\n return format(`(data->>'%I')::text ${operator} %L::text`, key, value);\n } else if (isDate(value)) {\n return format(\n `(data->>'%I')::timestamp ${operator} %L::timestamp`,\n key,\n value,\n );\n } else if (isNumber(value)) {\n return format(`(data->>'%I')::numeric ${operator} %s`, key, value);\n } else {\n return format(`(data->>'%I') ${operator} %L`, key, value);\n }\n};\n\nconst constructSimpleFilterQueryValue = (\n key: string,\n value: unknown,\n): string => {\n if (isUUID(value)) {\n return format('%L::text', value);\n } else if (isDate(value)) {\n return format('%L::timestamp', value);\n } else if (isNumber(value)) {\n return format('%L', value);\n } else {\n return format('%L', value);\n }\n};\n\nconst isUUID = (value: unknown): boolean => {\n return typeof value === 'string' && /^[0-9a-fA-F-]{36}$/.test(value);\n};\n\nconst isDate = (value: unknown): boolean => {\n return typeof value === 'string' && !isNaN(Date.parse(value));\n};\n\nconst isNumber = (value: unknown): boolean => {\n return typeof value === 'number';\n};\n","import pg from 'pg';\n\nconst pools: Map<string, pg.Pool> = new Map();\n\nexport const getPool = (\n connectionStringOrOptions: string | pg.PoolConfig,\n): pg.Pool => {\n const connectionString =\n typeof connectionStringOrOptions === 'string'\n ? connectionStringOrOptions\n : connectionStringOrOptions.connectionString!;\n\n const poolOptions =\n typeof connectionStringOrOptions === 'string'\n ? { connectionString }\n : connectionStringOrOptions;\n\n return (\n pools.get(connectionString) ??\n pools.set(connectionString, new pg.Pool(poolOptions)).get(connectionString)!\n );\n};\n\nexport const endPool = async (connectionString: string): Promise<void> => {\n const pool = pools.get(connectionString);\n if (pool) {\n await pool.end();\n pools.delete(connectionString);\n }\n};\n\nexport const endAllPools = () =>\n Promise.all(\n [...pools.keys()].map((connectionString) => endPool(connectionString)),\n );\n","import format from 'pg-format';\nimport type { PongoUpdate } from '../../main';\n\nexport const constructUpdateQuery = <T>(update: PongoUpdate<T>): string => {\n let updateQuery = 'data';\n\n if ('$set' in update) {\n const setUpdate = update.$set!;\n updateQuery = format(\n 'jsonb_set(%s, %L, data || %L)',\n updateQuery,\n '{}',\n JSON.stringify(setUpdate),\n );\n }\n\n if ('$unset' in update) {\n const unsetUpdate = Object.keys(update.$unset!);\n updateQuery = format('%s - %L', updateQuery, unsetUpdate.join(', '));\n }\n\n if ('$inc' in update) {\n const incUpdate = update.$inc!;\n for (const [key, value] of Object.entries(incUpdate)) {\n updateQuery = format(\n \"jsonb_set(%s, '{%s}', to_jsonb((data->>'%s')::numeric + %L))\",\n updateQuery,\n key,\n key,\n value,\n );\n }\n }\n\n if ('$push' in update) {\n const pushUpdate = update.$push!;\n for (const [key, value] of Object.entries(pushUpdate)) {\n updateQuery = format(\n \"jsonb_set(%s, '{%s}', (COALESCE(data->'%s', '[]'::jsonb) || to_jsonb(%L)))\",\n updateQuery,\n key,\n key,\n value,\n );\n }\n }\n\n return updateQuery;\n};\n","import { postgresClient } from '../postgres';\nimport type { PongoCollection } from './typing';\n\nexport interface DbClient {\n connect(): Promise<void>;\n close(): Promise<void>;\n collection: <T>(name: string) => PongoCollection<T>;\n}\n\nexport const getDbClient = (\n connectionString: string,\n database?: string,\n): DbClient => {\n // This is the place where in the future could come resolution of other database types\n return postgresClient(connectionString, database);\n};\n","import { getDbClient } from './dbClient';\nimport type { PongoClient, PongoDb } from './typing';\n\nexport const pongoClient = (connectionString: string): PongoClient => {\n const dbClient = getDbClient(connectionString);\n\n const pongoClient: PongoClient = {\n connect: async () => {\n await dbClient.connect();\n return pongoClient;\n },\n close: () => dbClient.close(),\n db: (dbName?: string): PongoDb =>\n dbName ? getDbClient(connectionString, dbName) : dbClient,\n };\n\n return pongoClient;\n};\n"],"mappings":"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","names":["uuid","format","sql","pool","sqlText","params","client","query","format","constructFilterQuery","filter","key","value","constructComplexFilterQuery","constructSimpleFilterQuery","isUUID","isDate","isNumber","operator","val","constructComparisonFilterQuery","v","constructSimpleFilterQueryValue","pg","pools","getPool","connectionStringOrOptions","connectionString","poolOptions","endPool","pool","endAllPools","format","constructUpdateQuery","update","updateQuery","setUpdate","unsetUpdate","incUpdate","key","value","pushUpdate","postgresClient","connectionString","database","pool","getPool","name","postgresCollection","collectionName","createCollection","sql","document","id","uuid","filter","update","filterQuery","constructFilterQuery","updateQuery","constructUpdateQuery","result","row","getDbClient","connectionString","database","postgresClient","pongoClient","connectionString","dbClient","getDbClient","dbName"]}
package/package.json ADDED
@@ -0,0 +1,61 @@
1
+ {
2
+ "name": "@event-driven-io/pongo",
3
+ "version": "0.0.1",
4
+ "description": "Pongo - Mongo with strong consistency on top of Postgres",
5
+ "type": "module",
6
+ "scripts": {
7
+ "build": "tsup",
8
+ "build:ts": "tsc",
9
+ "build:ts:watch": "tsc --watch",
10
+ "test": "run-s test:unit test:int test:e2e",
11
+ "test:unit": "glob -c \"node --import tsx --test\" **/*.unit.spec.ts",
12
+ "test:int": "glob -c \"node --import tsx --test\" **/*.int.spec.ts",
13
+ "test:e2e": "glob -c \"node --import tsx --test\" **/*.e2e.spec.ts",
14
+ "test:watch": "node --import tsx --test --watch",
15
+ "test:unit:watch": "glob -c \"node --import tsx --test --watch\" **/*.unit.spec.ts",
16
+ "test:int:watch": "glob -c \"node --import tsx --test --watch\" **/*.int.spec.ts",
17
+ "test:e2e:watch": "glob -c \"node --import tsx --test --watch\" **/*.e2e.spec.ts"
18
+ },
19
+ "repository": {
20
+ "type": "git",
21
+ "url": "git+https://github.com/event-driven-io/Pongo.git"
22
+ },
23
+ "keywords": [
24
+ "Event Sourcing"
25
+ ],
26
+ "author": "Oskar Dudycz",
27
+ "bugs": {
28
+ "url": "https://github.com/event-driven-io/Pongo/issues"
29
+ },
30
+ "homepage": "https://event-driven-io.github.io/Pongo/",
31
+ "exports": {
32
+ ".": {
33
+ "import": {
34
+ "types": "./dist/index.d.ts",
35
+ "default": "./dist/index.js"
36
+ },
37
+ "require": {
38
+ "types": "./dist/index.d.cts",
39
+ "default": "./dist/index.cjs"
40
+ }
41
+ }
42
+ },
43
+ "main": "./dist/index.cjs",
44
+ "module": "./dist/index.js",
45
+ "types": "./dist/index.d.ts",
46
+ "files": [
47
+ "dist"
48
+ ],
49
+ "peerDependencies": {
50
+ "@types/uuid": "^9.0.8",
51
+ "close-with-grace": "^1.3.0",
52
+ "pg": "^8.12.0",
53
+ "pg-format": "^1.0.4",
54
+ "uuid": "^9.0.1"
55
+ },
56
+ "devDependencies": {
57
+ "@types/node": "20.11.30",
58
+ "@types/pg": "^8.11.6",
59
+ "@types/pg-format": "^1.0.5"
60
+ }
61
+ }