@prisma/dev 0.0.0-dev.202505221232 → 0.0.0-dev.202505221432

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.
Files changed (2) hide show
  1. package/dist/index.js +5 -5
  2. package/package.json +2 -1
package/dist/index.js CHANGED
@@ -1,8 +1,8 @@
1
1
  import { createRequire } from 'module'; const require = createRequire(import.meta.url);
2
- import{createServer as Ge}from"http";import{promisify as ze}from"util";import{serve as Je}from"@hono/node-server";import We from"@prisma/get-platform";import{logger as Qe}from"hono/logger";import{Hono as ie}from"hono/tiny";import{validator as w}from"hono/validator";import{process as Xe}from"std-env";import{HTTPException as P}from"hono/http-exception";import{object as me,optional as le,parseJson as ge,pipe as R,regex as C,safeParse as he,string as O,url as L}from"valibot";var M=/^(postgres|postgresql):\/\//,fe=R(O(),ge(),me({databaseUrl:R(O(),L(),C(M)),shadowDatabaseUrl:le(R(O(),L(),C(M)))}));function j(t){return Buffer.from(JSON.stringify(t),"utf8").toString("base64url")}function ye(t){let e=Buffer.from(t,"base64url").toString("utf8"),{issues:r,output:n,success:o}=he(fe,e,{abortEarly:!0});return o?[null,n]:[r]}var m=(t,e)=>{let{authorization:r}=t;if(!r)throw new P(401,{message:"Missing API Key"});let[n,o="",s]=r.split(" ");if(n!=="Bearer"||s)throw new P(401,{message:"Invalid API Key"});let[a,i]=ye(o);if(a)throw new P(401,{message:"Invalid API Key",cause:a.join(", ")});let{databaseUrl:c}=i;if(c!==e.var.db.connectionString)throw new P(401,{message:"Unauthorized"});return{decodedAPIKey:i}};import{array as be,literal as we,minLength as Se,object as ve,pipe as Pe,safeParse as Ee,string as Ae,union as Te}from"valibot";var De=ve({tags:Te([Pe(be(Ae()),Se(1)),we("all")])});async function U(t){let{output:e,success:r}=Ee(De,await t.req.json(),{abortEarly:!0});return r?e:t.text("Invalid input",400)}import{spawn as ke}from"child_process";import{once as Be}from"events";import{mkdir as Ce}from"fs/promises";import{join as Le}from"path";import{process as Me}from"std-env";import{createWriteStream as Re,WriteStream as Oe}from"fs";import{chmod as xe,stat as Ie,writeFile as $e}from"fs/promises";import _e from"env-paths";import{inflate as He}from"pako";var F=_e("prisma-dev");function N(t,e){return`${F.cache}/engine/${t}/${e}`}function q(t){return`${F.data}/${t}`}async function V(t){try{return await Ie(t),!0}catch(e){if(e!=null&&typeof e=="object"&&"code"in e&&e.code==="ENOENT")return!1;throw e}}async function K(t,e){let r=He(t);await $e(e,r),await xe(e,"755")}async function G(t,e){await t.stream().pipeTo(Oe.toWeb(Re(e,{encoding:"utf-8"})))}function E(){let t,e,r=new Promise((n,o)=>{t=n,e=o});return{fail:e,promise:r,succeed:t}}var y=class t{static#r=new Map;#e;#t;constructor(e){this.#e=e,this.#t=null}static async get(e){let r=`${e.schemaHash}:${e.clientVersion}`;try{let n=t.#r.get(r);if(n)return n;let o=new t(e);return t.#r.set(r,o),e.debug&&console.debug("starting engine...",e),await o.start(),e.debug&&console.debug("engine started!"),o}finally{t.stopAll(r)}}static async stopAll(e){let n=(await Promise.allSettled(Array.from(t.#r.entries()).filter(([o])=>o!==e).map(async([o,s])=>{try{await s.stop()}finally{t.#r.delete(o)}}))).filter(o=>o.status==="rejected").map(o=>o.reason);if(n.length>0)throw new AggregateError(n,"Failed to stop engines")}async commitTransaction(e,r){return await this.#o(e,r,"commit")}async request(e,r){let{url:n}=await this.#t,o=this.#n(r),s=await fetch(n,{body:typeof e=="string"?e:JSON.stringify(e),headers:{...o,"Content-Type":"application/json"},method:"POST"});if(!s.ok)throw await l.fromResponse(s);return await s.text()}async rollbackTransaction(e,r){return await this.#o(e,r,"rollback")}async startTransaction(e,r){let{url:n}=await this.#t,o=this.#n(r),s=await fetch(`${n}/transaction/start`,{body:JSON.stringify(e),headers:{...o,"Content-Type":"application/json"},method:"POST"});if(!s.ok)throw await l.fromResponse(s);return await s.json()}async start(){if(this.#t!=null)return;let e=await this.#s();this.#e.debug&&console.debug("spinning up engine at path...",e);let r=ke(e,["--enable-raw-queries","--enable-telemetry-in-response","--port","0"],{env:{LOG_QUERIES:"y",PRISMA_DML:this.#e.base64Schema,QE_LOG_LEVEL:"TRACE",RUST_BACKTRACE:"1",RUST_LOG:"info"},stdio:["ignore","pipe","pipe"],windowsHide:!0});r.stderr.setEncoding("utf8"),r.stdout.setEncoding("utf8");let{fail:n,promise:o,succeed:s}=E();this.#t=o;let a=i=>{let c=i.split(`
3
- `).find(pe=>pe.includes("Started query engine http server"));if(!c)return;r.stdout.removeListener("data",a);let{fields:f}=JSON.parse(c);if(f==null)return n(new Error(`Unexpected data during initialization, "fields" are missing: ${i}`));let{ip:p,port:u}=f;if(p==null||u==null)return n(new Error(`This version of query-engine is not compatible with minippg, "ip" and "port" are missing in the startup log entry.
4
- Received data: ${i}`));s({childProcess:r,url:`http://${p}:${u}`})};r.once("error",i=>{this.#t=null,n(new b(String(i)))}),r.once("exit",(i,c)=>{this.#t=null,n(new b(`Query Engine exited with code ${i} and signal ${c}`))}),r.stdout.on("data",a),r.stderr.on("data",i=>{console.error(i)}),await this.#t}async stop(){if(this.#t==null)return;let{childProcess:e}=await this.#t;e.exitCode==null&&e.signalCode==null&&(e.kill(),await Be(e,"exit"))}async#s(){this.#e.debug&&console.debug("getting engine commit hash...");let e=await this.#a();this.#e.debug&&console.debug("got engine commit hash",e);let r=N(this.#e.clientVersion,e);this.#e.debug&&console.debug("cache directory path",r),await Ce(r,{recursive:!0});let{platform:n}=this.#e.platform,o=n==="windows"?".exe":"",s=Le(r,`query-engine-${n}${o}`);return this.#e.debug&&console.debug("engine binary path",s),(Me.env.PRISMA_DEV_FORCE_ENGINE_DOWNLOAD==="1"||await V(s)===!1)&&await this.#i({commitHash:e,extension:o,engineBinaryPath:s}),s}async#a(){let e=await fetch(`https://registry.npmjs.org/@prisma/client/${this.#e.clientVersion}`);if(!e.ok)throw new Error(`Couldn't fetch package.json from npm registry, status code: ${e.status}`);let n=(await e.json()).devDependencies?.["@prisma/engines-version"];if(!n)throw new Error("Couldn't find engines version in package.json");let o=n.split(".").at(-1);if(!o)throw new Error("Couldn't find commit hash in engines version");return o}async#i(e){let{commitHash:r,extension:n,engineBinaryPath:o}=e,{binaryTarget:s}=this.#e.platform,a=`https://binaries.prisma.sh/all_commits/${r}/${s}/query-engine${n}.gz`;this.#e.debug&&console.debug("downloading engine from url",a);let i=await fetch(a);if(!i.ok)throw new Error(`Couldn't download engine. URL: ${a}, status code: ${i.status}`);await K(await i.arrayBuffer(),o),this.#e.debug&&console.debug("engine downloaded and saved at",o)}#n(e){let r={};for(let[n,o]of Object.entries(e))o!=null&&(r[n]=o);return r}async#o(e,r,n){let{url:o}=await this.#t,s=this.#n(r),a=await fetch(`${o}/transaction/${e}/${n}`,{headers:{...s,"Content-Type":"application/json"},method:"POST"});if(!a.ok)throw await l.fromResponse(a);try{return await a.json()}catch{return{}}}};function A(t,e){return console.error(t),t instanceof b?e.json({EngineNotStarted:{reason:{EngineStartupError:{logs:[],msg:t.message}}}},500):t instanceof l?e.text(t.responseBody,t.statusCode):e.body(null,500)}var b=class extends Error{name="EngineStartError"},l=class t extends Error{constructor(r,n,o){super(`${r}: Query Engine response status ${n}, body: ${o}`);this.action=r;this.statusCode=n;this.responseBody=o}name="EngineHttpError";static async fromResponse(r){let n=new URL(r.url),o=await r.text();return new t(n.pathname,r.status,o)}};var z=51214,J=51213,W=51215,g=class extends Error{constructor(r,n){super(`Port number \`${r}\` is not available for service ${n}.`);this.port=r;this.service=n}name="PortNotAvailableError"};import{Buffer as Q}from"buffer";var T=new Map;async function x(t){let r=new TextEncoder().encode(t),n=await crypto.subtle.digest("SHA-256",r);return Array.from(new Uint8Array(n)).map(a=>a.toString(16).padStart(2,"0")).join("")}function X(t){let e=t.req.param("schemaHash"),r=T.get(e);return r==null?t.json({EngineNotStarted:{reason:"SchemaMissing"}},404):{schemaHash:e,schemas:r}}var je=/datasource\s+db\s+\{\s*provider\s*=\s*"postgres(!?ql)?"\s+url\s*=\s*.+\s*\}/;async function Y(t,e){let r=Q.from(t,"base64").toString("utf8"),n=`datasource db {
2
+ import{createServer as et}from"http";import{promisify as tt}from"util";import{serve as rt}from"@hono/node-server";import nt from"@prisma/get-platform";import{logger as ot}from"hono/logger";import{Hono as pe}from"hono/tiny";import{validator as S}from"hono/validator";import{process as st}from"std-env";import{HTTPException as E}from"hono/http-exception";import{object as he,optional as be,parseJson as ye,pipe as x,regex as U,safeParse as we,string as I,url as N}from"valibot";var q=/^(postgres|postgresql):\/\//,Se=x(I(),ye(),he({databaseUrl:x(I(),N(),U(q)),shadowDatabaseUrl:be(x(I(),N(),U(q)))}));function V(t){return Buffer.from(JSON.stringify(t),"utf8").toString("base64url")}function Pe(t){let e=Buffer.from(t,"base64url").toString("utf8"),{issues:r,output:n,success:o}=we(Se,e,{abortEarly:!0});return o?[null,n]:[r]}var p=(t,e)=>{let{authorization:r}=t;if(!r)throw new E(401,{message:"Missing API Key"});let[n,o="",s]=r.split(" ");if(n!=="Bearer"||s)throw new E(401,{message:"Invalid API Key"});let[a,i]=Pe(o);if(a)throw new E(401,{message:"Invalid API Key",cause:a.join(", ")});let{databaseUrl:c}=i;if(c!==e.var.db.connectionString)throw new E(401,{message:"Unauthorized"});return{decodedAPIKey:i}};import{array as ve,literal as Ee,minLength as De,object as Ae,pipe as Te,safeParse as Re,string as Oe,union as xe}from"valibot";var Ie=Ae({tags:xe([Te(ve(Oe()),De(1)),Ee("all")])});async function K(t){let{output:e,success:r}=Re(Ie,await t.req.json(),{abortEarly:!0});return r?e:t.text("Invalid input",400)}import{spawn as Ne}from"child_process";import{once as qe}from"events";import{mkdir as Ve}from"fs/promises";import{join as Ke}from"path";import{process as Ge}from"std-env";import{createWriteStream as $e,WriteStream as ke}from"fs";import{access as He,chmod as _e,constants as Be,readFile as Ce,stat as Le,unlink as Me,writeFile as Fe}from"fs/promises";import je from"env-paths";import{inflate as Ue}from"pako";var G=je("prisma-dev");function z(t,e){return`${G.cache}/engine/${t}/${e}`}function J(t){return`${G.data}/${t}`}async function D(t){try{return await He(t,Be.F_OK),!0}catch(e){if(k(e))return!1;throw e}}async function Q(t,e){let r=Ue(t);await Fe(e,r),await _e(e,"755")}async function W(t,e){await t.stream().pipeTo(ke.toWeb($e(e,{encoding:"utf-8"})))}async function X(t){try{return await Me(t),!0}catch{return!1}}async function $(t){try{return(await Le(t)).mtimeMs}catch(e){if(k(e))return-1/0;throw e}}function k(t){return t!=null&&typeof t=="object"&&"code"in t&&t.code==="ENOENT"}async function Y(t){try{return await Ce(t,{encoding:"utf-8"})}catch(e){if(k(e))return null;throw e}}function A(){let t,e,r=new Promise((n,o)=>{t=n,e=o});return{fail:e,promise:r,succeed:t}}var y=class t{static#r=new Map;#e;#t;constructor(e){this.#e=e,this.#t=null}static async get(e){let r=`${e.schemaHash}:${e.clientVersion}`;try{let n=t.#r.get(r);if(n)return n;let o=new t(e);return t.#r.set(r,o),e.debug&&console.debug("starting engine...",e),await o.start(),e.debug&&console.debug("engine started!"),o}finally{t.stopAll(r)}}static async stopAll(e){let n=(await Promise.allSettled(Array.from(t.#r.entries()).filter(([o])=>o!==e).map(async([o,s])=>{try{await s.stop()}finally{t.#r.delete(o)}}))).filter(o=>o.status==="rejected").map(o=>o.reason);if(n.length>0)throw new AggregateError(n,"Failed to stop engines")}async commitTransaction(e,r){return await this.#i(e,r,"commit")}async request(e,r){let{url:n}=await this.#t,o=this.#n(r),s=await fetch(n,{body:typeof e=="string"?e:JSON.stringify(e),headers:{...o,"Content-Type":"application/json"},method:"POST"});if(!s.ok)throw await g.fromResponse(s);return await s.text()}async rollbackTransaction(e,r){return await this.#i(e,r,"rollback")}async startTransaction(e,r){let{url:n}=await this.#t,o=this.#n(r),s=await fetch(`${n}/transaction/start`,{body:JSON.stringify(e),headers:{...o,"Content-Type":"application/json"},method:"POST"});if(!s.ok)throw await g.fromResponse(s);return await s.json()}async start(){if(this.#t!=null)return;let e=await this.#s();this.#e.debug&&console.debug("spinning up engine at path...",e);let r=Ne(e,["--enable-raw-queries","--enable-telemetry-in-response","--port","0"],{env:{LOG_QUERIES:"y",PRISMA_DML:this.#e.base64Schema,QE_LOG_LEVEL:"TRACE",RUST_BACKTRACE:"1",RUST_LOG:"info"},stdio:["ignore","pipe","pipe"],windowsHide:!0});r.stderr.setEncoding("utf8"),r.stdout.setEncoding("utf8");let{fail:n,promise:o,succeed:s}=A();this.#t=o;let a=i=>{let c=i.split(`
3
+ `).find(u=>u.includes("Started query engine http server"));if(!c)return;r.stdout.removeListener("data",a);let{fields:l}=JSON.parse(c);if(l==null)return n(new Error(`Unexpected data during initialization, "fields" are missing: ${i}`));let{ip:b,port:m}=l;if(b==null||m==null)return n(new Error(`This version of query-engine is not compatible with minippg, "ip" and "port" are missing in the startup log entry.
4
+ Received data: ${i}`));s({childProcess:r,url:`http://${b}:${m}`})};r.once("error",i=>{this.#t=null,n(new w(String(i)))}),r.once("exit",(i,c)=>{this.#t=null,n(new w(`Query Engine exited with code ${i} and signal ${c}`))}),r.stdout.on("data",a),r.stderr.on("data",i=>{console.error(i)}),await this.#t}async stop(){if(this.#t==null)return;let{childProcess:e}=await this.#t;e.exitCode==null&&e.signalCode==null&&(e.kill(),await qe(e,"exit"))}async#s(){this.#e.debug&&console.debug("getting engine commit hash...");let e=await this.#o();this.#e.debug&&console.debug("got engine commit hash",e);let r=z(this.#e.clientVersion,e);this.#e.debug&&console.debug("cache directory path",r),await Ve(r,{recursive:!0});let{platform:n}=this.#e.platform,o=n==="windows"?".exe":"",s=Ke(r,`query-engine-${n}${o}`);return this.#e.debug&&console.debug("engine binary path",s),(Ge.env.PRISMA_DEV_FORCE_ENGINE_DOWNLOAD==="1"||await D(s)===!1)&&await this.#a({commitHash:e,extension:o,engineBinaryPath:s}),s}async#o(){let e=await fetch(`https://registry.npmjs.org/@prisma/client/${this.#e.clientVersion}`);if(!e.ok)throw new Error(`Couldn't fetch package.json from npm registry, status code: ${e.status}`);let n=(await e.json()).devDependencies?.["@prisma/engines-version"];if(!n)throw new Error("Couldn't find engines version in package.json");let o=n.split(".").at(-1);if(!o)throw new Error("Couldn't find commit hash in engines version");return o}async#a(e){let{commitHash:r,extension:n,engineBinaryPath:o}=e,{binaryTarget:s}=this.#e.platform,a=`https://binaries.prisma.sh/all_commits/${r}/${s}/query-engine${n}.gz`;this.#e.debug&&console.debug("downloading engine from url",a);let i=await fetch(a);if(!i.ok)throw new Error(`Couldn't download engine. URL: ${a}, status code: ${i.status}`);await Q(await i.arrayBuffer(),o),this.#e.debug&&console.debug("engine downloaded and saved at",o)}#n(e){let r={};for(let[n,o]of Object.entries(e))o!=null&&(r[n]=o);return r}async#i(e,r,n){let{url:o}=await this.#t,s=this.#n(r),a=await fetch(`${o}/transaction/${e}/${n}`,{headers:{...s,"Content-Type":"application/json"},method:"POST"});if(!a.ok)throw await g.fromResponse(a);try{return await a.json()}catch{return{}}}};function T(t,e){return console.error(t),t instanceof w?e.json({EngineNotStarted:{reason:{EngineStartupError:{logs:[],msg:t.message}}}},500):t instanceof g?e.text(t.responseBody,t.statusCode):e.body(null,500)}var w=class extends Error{name="EngineStartError"},g=class t extends Error{constructor(r,n,o){super(`${r}: Query Engine response status ${n}, body: ${o}`);this.action=r;this.statusCode=n;this.responseBody=o}name="EngineHttpError";static async fromResponse(r){let n=new URL(r.url),o=await r.text();return new t(n.pathname,r.status,o)}};var Z=51214,ee=51213,te=51215,f=class extends Error{constructor(r,n){super(`Port number \`${r}\` is not available for service ${n}.`);this.port=r;this.service=n}name="PortNotAvailableError"};import{Buffer as re}from"buffer";var R=new Map;async function H(t){let r=new TextEncoder().encode(t),n=await crypto.subtle.digest("SHA-256",r);return Array.from(new Uint8Array(n)).map(a=>a.toString(16).padStart(2,"0")).join("")}function ne(t){let e=t.req.param("schemaHash"),r=R.get(e);return r==null?t.json({EngineNotStarted:{reason:"SchemaMissing"}},404):{schemaHash:e,schemas:r}}var ze=/datasource\s+db\s+\{\s*provider\s*=\s*"postgres(!?ql)?"\s+url\s*=\s*.+\s*\}/;async function oe(t,e){let r=re.from(t,"base64").toString("utf8"),n=`datasource db {
5
5
  provider = "postgresql"
6
6
  url = "${e}"
7
- }`,o=r.replace(je,n),s=await x(o);return{base64Override:Q.from(o,"utf8").toString("base64"),overrideHash:s}}function D(t){let{req:e}=t;return{traceparent:e.header("traceparent"),"X-capture-telemetry":e.header("X-capture-telemetry")}}import{integer as Z,looseObject as Ue,minValue as ee,number as I,object as Fe,optional as Ne,pipe as te,safeParse as re,string as ne,union as qe}from"valibot";var Ve=Fe({isolation_level:Ne(ne()),max_wait:te(I(),Z(),ee(0)),timeout:te(I(),Z(),ee(0))});async function oe(t){let{issues:e,output:r,success:n}=re(Ve,await t.req.json(),{abortEarly:!0});return n?r:t.json({EngineNotStarted:{reason:"InvalidRequest",issues:e}},400)}var Ke=Ue({id:qe([ne(),I()])});function se(t,e){let{output:r,success:n}=re(Ke,t);return n?r:e.json({EngineMalfunction:{}},500)}async function ce(t,e){let{port:r}=e;if(e.dryRun)return ae(r,null);let n=await Ye(r,t,e),{fail:o,promise:s,succeed:a}=E(),i=Je({createServer:Ge,fetch:n.fetch,port:r},()=>{a()});return i.on("error",c=>{if(typeof c=="object"&&"code"in c&&c.code==="EADDRINUSE")return o(new g(r,"server"));console.error(c)}),await s,ae(r,i)}function ae(t,e){return{async close(){e&&await Promise.allSettled([ze(e.close.bind(e))(),y.stopAll()])},port:t,url:`http://localhost:${t}`}}async function Ye(t,e,r){let{debug:n}=r,o=new ie,s=await We.getPlatformInfo();return n&&console.debug("platform info: %s",JSON.stringify(s)),n&&o.use("*",Qe()),o.use("*",async(a,i)=>(a.set("db",e),a.set("debug",!!n),a.set("platform",s),a.set("port",t),a.set("protocol","http"),a.set("serverState",r),await i())),o.route("/",S),o}var S=new ie;S.get("/health",t=>t.json({name:t.get("serverState").name}));S.post("/invalidate",w("header",m),async t=>{let e=await U(t);return e instanceof Response?e:t.body(null)});var Ze="/:clientVersion/:schemaHash",v=S.basePath(Ze);S.route("/",v);var et=["/graphql","/itx/:transactionId/graphql"];v.on("POST",[...et],w("header",m),async t=>{let{req:e}=t;try{let r=await $(t);if(r instanceof Response)return r;let n=await e.text(),o=e.param("transactionId"),s=await r.request(n,{...D(t),"X-transaction-id":o});return t.text(s)}catch(r){return A(r,t)}});v.basePath("/itx/:transactionId").on("POST",["/commit","/rollback"],w("header",m),async t=>{let{req:e}=t;try{let r=await $(t);if(r instanceof Response)return r;let o=`${e.routePath.split("/").filter(Boolean).at(-1)}Transaction`,s=e.param("transactionId"),a=await r[o](s,D(t));return t.json(a)}catch(r){return A(r,t)}});v.put("/schema",w("header",m),async t=>{let{req:e}=t,r=await e.text();if(!r)return t.text("Missing schema",400);let n=e.param("schemaHash"),o=T.get(n);if(o==null){if(n!==await x(r))return t.text("Schema hash mismatch",400);let s=await Y(r,t.get("db").connectionString);return T.set(n,{base64Original:r,...s}),t.text(n)}return r!==o.base64Original?t.text("Schema mismatch",400):t.text(n)});v.post("/transaction/start",w("header",m),async t=>{let{req:e}=t,r=await oe(t);if(r instanceof Response)return r;try{let n=await $(t);if(n instanceof Response)return n;let o=await n.startTransaction(r,D(t)),s=se(o,t);if(s instanceof Response)return s;let{id:a}=s,i=e.param("clientVersion"),c=t.get("port"),f=t.get("protocol"),p=e.param("schemaHash");return t.json({...o,"data-proxy":{endpoint:`${f}://localhost:${c}/${i}/${p}/itx/${a}`}})}catch(n){return A(n,t)}});async function $(t){let{req:e}=t,r=X(t);if(r instanceof Response)return r;let{base64Override:n,overrideHash:o}=r.schemas;return await y.get({base64Schema:n,clientVersion:Xe.env.PRISMA_DEV_FORCE_CLIENT_VERSION||e.param("clientVersion"),debug:t.get("debug"),platform:t.get("platform"),schemaHash:o})}import{PGlite as tt}from"@electric-sql/pglite";import{PGLiteSocketServer as rt}from"@electric-sql/pglite-socket";import{pgDump as nt}from"@electric-sql/pglite-tools/pg_dump";var d={connectionLimit:1,connectTimeout:0,database:"postgres",maxIdleConnectionLifetime:0,password:"postgres",poolTimeout:0,socketTimeout:0,sslMode:"disable",username:"postgres"},ot=`postgres://${d.username}:${d.password}@localhost`,st=new URLSearchParams({connection_limit:String(d.connectionLimit),connect_timeout:String(d.connectTimeout),max_idle_connection_lifetime:String(d.maxIdleConnectionLifetime),pool_timeout:String(d.poolTimeout),socket_timeout:String(d.socketTimeout),sslmode:d.sslMode});async function _(t,e){let r=t==="database"?e.databasePort:e.shadowDatabasePort;if(e.dryRun)return de(t,e,{db:null,port:r,server:null});let{debug:n}=e,o=await tt.create({database:d.database,dataDir:"memory://",debug:n?5:void 0,defaultDataTransferContainer:"file",username:d.username});await o.waitReady,t!=="shadow_database"&&await at(o,e);let s=new rt({db:o,inspect:n,port:r});try{await s.start()}catch(a){throw a instanceof Error&&"code"in a&&a.code==="EADDRINUSE"?new g(r,t):a}return de(t,e,{db:o,port:r,server:s})}function de(t,e,r){let{debug:n}=e,{db:o,port:s,server:a}=r||{};return n&&console.debug(`${t} server started on port %i`,s),{...d,close:async()=>{let i=[];try{await a?.stop(),n&&console.debug(`${t} server stopped on port %i`,s)}catch(c){console.error("server stop error",c),i.push(c)}if(t!=="shadow_database")try{await it(o,e),n&&console.debug(`${t} state saved`)}catch(c){console.error("dump error",c),i.push(c)}try{await o?.close(),n&&console.debug(`${t} closed`)}catch(c){console.error("db close error",c),i.push(c)}if(i.length>0)throw new AggregateError(i,`Failed to close ${t} properly`)},connectionString:ct(s),port:s}}async function at(t,e){try{let r=await e.readDatabaseDump();if(!r.length)return;await t.exec(`${r}; set search_path to public;`)}catch(r){if(r instanceof Error&&"code"in r&&r.code==="ENOENT")return console.warn("No database dump found. Starting with an empty database.");throw r}}async function it(t,e){await e.writeDatabaseDump(()=>nt({args:[],pg:t}))}function ct(t){return`${ot}:${t}/${d.database}?${st.toString()}`}import{copyFile as dt,mkdir as ut,readFile as pt,unlink as mt,writeFile as lt}from"fs/promises";import{join as ue}from"pathe";import{lock as gt}from"proper-lockfile";import{process as ht}from"std-env";var B=Symbol("initialize"),h=class{databasePort;debug;dryRun;name;persistenceMode;port;shadowDatabasePort;constructor(e){this.databasePort=e.databasePort??51214,this.debug=e.debug??!1,this.dryRun=e.dryRun??!1,this.name=e.name??"default",this.persistenceMode=e.persistenceMode,this.port=e.port??51213,this.shadowDatabasePort=e.shadowDatabasePort??51215}static async get(e){let r=e?.dryRun!==!0&&e?.persistenceMode!=="stateless"?new k(e):new H(e);return await r[B](),r}},H=class extends h{constructor(e){super({...e,persistenceMode:"stateless"})}async[B](){}readDatabaseDump(){return Promise.resolve("")}async writeDatabaseDump(){}async writeServerDump(){}},k=class extends h{#r;#e;#t;constructor(e){super({...e,persistenceMode:"stateful"}),this.#r=q(this.name),this.#e=ue(this.#r,"ppg.sql"),this.#t=ue(this.#r,"prisma-dev.json")}async[B](){await ut(this.#r,{recursive:!0}),this.debug&&console.debug(`using data directory: ${this.#r}`);try{await gt(this.#r,{lockfilePath:`${this.#t.replace("json","lock")}`}),this.debug&&console.debug(`obtained lock on: ${this.#r}`),await this.writeServerDump()}catch(e){throw e instanceof Error&&"code"in e&&e.code==="ELOCKED"?new Error(`A server with the name "${this.name}" is already running. `):e}}async readDatabaseDump(){return await pt(this.#e,{encoding:"utf-8"})}async writeDatabaseDump(e){let r=!1,n=`${this.#e}.bak`;try{await dt(this.#e,n),r=!0,this.debug&&console.debug(`backed up dump to: ${n}`)}catch(s){if(!(s instanceof Error&&"code"in s&&s.code==="ENOENT"))throw s}let o=await e();if(await G(o,this.#e),this.debug&&console.debug(`dumped database to: ${this.#e}`),!!r)try{await mt(`${this.#e}.bak`),this.debug&&console.log(`removed backup dump: ${n}`)}catch{}}async writeServerDump(e){await lt(this.#t,`${JSON.stringify({name:this.name,pid:ht.pid,port:this.port,databasePort:this.databasePort,shadowDatabasePort:this.shadowDatabasePort,exports:e},null,2)}
8
- `,{encoding:"utf-8"})}};async function Pr(t){let e=await h.get(t),[r,n]=await Promise.all([_("database",e),_("shadow_database",e)]),o=await ce(r,e),s=`prisma+postgres://localhost:${o.port}/?${new URLSearchParams({api_key:j({databaseUrl:r.connectionString,shadowDatabaseUrl:n.connectionString})}).toString()}`,a={accelerate:{url:o.url},database:{connectionString:r.connectionString},ppg:{url:s},shadowDatabase:{connectionString:n.connectionString}};return await e.writeServerDump(a),{...a,close:()=>i([o,r,n])};async function i(c){let p=(await Promise.allSettled(c.map(u=>u.close()))).filter(u=>u.status==="rejected").map(u=>new Error(u.reason));if(p.length>0)throw new AggregateError(p,"Failed to close some servers")}}export{z as DEFAULT_DATABASE_PORT,J as DEFAULT_SERVER_PORT,W as DEFAULT_SHADOW_DATABASE_PORT,g as PortNotAvailableError,Pr as unstable_startServer};
7
+ }`,o=r.replace(ze,n),s=await H(o);return{base64Override:re.from(o,"utf8").toString("base64"),overrideHash:s}}function O(t){let{req:e}=t;return{traceparent:e.header("traceparent"),"X-capture-telemetry":e.header("X-capture-telemetry")}}import{integer as se,looseObject as Je,minValue as ae,number as _,object as Qe,optional as We,pipe as ie,safeParse as ce,string as de,union as Xe}from"valibot";var Ye=Qe({isolation_level:We(de()),max_wait:ie(_(),se(),ae(0)),timeout:ie(_(),se(),ae(0))});async function ue(t){let{issues:e,output:r,success:n}=ce(Ye,await t.req.json(),{abortEarly:!0});return n?r:t.json({EngineNotStarted:{reason:"InvalidRequest",issues:e}},400)}var Ze=Je({id:Xe([de(),_()])});function le(t,e){let{output:r,success:n}=ce(Ze,t);return n?r:e.json({EngineMalfunction:{}},500)}async function ge(t,e){let{port:r}=e;if(e.dryRun)return me(r,null);let n=await at(r,t,e),{fail:o,promise:s,succeed:a}=A(),i=rt({createServer:et,fetch:n.fetch,port:r},()=>{a()});return i.on("error",c=>{if(typeof c=="object"&&"code"in c&&c.code==="EADDRINUSE")return o(new f(r,"server"));console.error(c)}),await s,me(r,i)}function me(t,e){return{async close(){e&&await Promise.allSettled([tt(e.close.bind(e))(),y.stopAll()])},port:t,url:`http://localhost:${t}`}}async function at(t,e,r){let{debug:n}=r,o=new pe,s=await nt.getPlatformInfo();return n&&console.debug("platform info: %s",JSON.stringify(s)),n&&o.use("*",ot()),o.use("*",async(a,i)=>(a.set("db",e),a.set("debug",!!n),a.set("platform",s),a.set("port",t),a.set("protocol","http"),a.set("serverState",r),await i())),o.route("/",P),o}var P=new pe;P.get("/health",t=>t.json({name:t.get("serverState").name}));P.post("/invalidate",S("header",p),async t=>{let e=await K(t);return e instanceof Response?e:t.body(null)});var it="/:clientVersion/:schemaHash",v=P.basePath(it);P.route("/",v);var ct=["/graphql","/itx/:transactionId/graphql"];v.on("POST",[...ct],S("header",p),async t=>{let{req:e}=t;try{let r=await B(t);if(r instanceof Response)return r;let n=await e.text(),o=e.param("transactionId"),s=await r.request(n,{...O(t),"X-transaction-id":o});return t.text(s)}catch(r){return T(r,t)}});v.basePath("/itx/:transactionId").on("POST",["/commit","/rollback"],S("header",p),async t=>{let{req:e}=t;try{let r=await B(t);if(r instanceof Response)return r;let o=`${e.routePath.split("/").filter(Boolean).at(-1)}Transaction`,s=e.param("transactionId"),a=await r[o](s,O(t));return t.json(a)}catch(r){return T(r,t)}});v.put("/schema",S("header",p),async t=>{let{req:e}=t,r=await e.text();if(!r)return t.text("Missing schema",400);let n=e.param("schemaHash"),o=R.get(n);if(o==null){if(n!==await H(r))return t.text("Schema hash mismatch",400);let s=await oe(r,t.get("db").connectionString);return R.set(n,{base64Original:r,...s}),t.text(n)}return r!==o.base64Original?t.text("Schema mismatch",400):t.text(n)});v.post("/transaction/start",S("header",p),async t=>{let{req:e}=t,r=await ue(t);if(r instanceof Response)return r;try{let n=await B(t);if(n instanceof Response)return n;let o=await n.startTransaction(r,O(t)),s=le(o,t);if(s instanceof Response)return s;let{id:a}=s,i=e.param("clientVersion"),c=t.get("port"),l=t.get("protocol"),b=e.param("schemaHash");return t.json({...o,"data-proxy":{endpoint:`${l}://localhost:${c}/${i}/${b}/itx/${a}`}})}catch(n){return T(n,t)}});async function B(t){let{req:e}=t,r=ne(t);if(r instanceof Response)return r;let{base64Override:n,overrideHash:o}=r.schemas;return await y.get({base64Schema:n,clientVersion:st.env.PRISMA_DEV_FORCE_CLIENT_VERSION||e.param("clientVersion"),debug:t.get("debug"),platform:t.get("platform"),schemaHash:o})}import{PGlite as dt}from"@electric-sql/pglite";import{PGLiteSocketServer as ut}from"@electric-sql/pglite-socket";import{pgDump as lt}from"@electric-sql/pglite-tools/pg_dump";var d={connectionLimit:1,connectTimeout:0,database:"postgres",maxIdleConnectionLifetime:0,password:"postgres",poolTimeout:0,socketTimeout:0,sslMode:"disable",username:"postgres"},mt=`postgres://${d.username}:${d.password}@localhost`,pt=new URLSearchParams({connection_limit:String(d.connectionLimit),connect_timeout:String(d.connectTimeout),max_idle_connection_lifetime:String(d.maxIdleConnectionLifetime),pool_timeout:String(d.poolTimeout),socket_timeout:String(d.socketTimeout),sslmode:d.sslMode});async function C(t,e){let r=t==="database"?e.databasePort:e.shadowDatabasePort;if(e.dryRun)return fe(t,e,{db:null,port:r,server:null});let{debug:n}=e,o=await dt.create({database:d.database,dataDir:"memory://",debug:n?5:void 0,defaultDataTransferContainer:"file",username:d.username});await o.waitReady,t==="database"&&await gt(o,e);let s=new ut({db:o,inspect:n,port:r});try{await s.start()}catch(a){throw a instanceof Error&&"code"in a&&a.code==="EADDRINUSE"?new f(r,t):a}return fe(t,e,{db:o,port:r,server:s})}function fe(t,e,r){let{debug:n}=e,{db:o,port:s,server:a}=r||{};return n&&console.debug(`${t} server started on port %i`,s),{...d,close:async()=>{let i=[];try{await a?.stop(),n&&console.debug(`${t} server stopped on port %i`,s)}catch(c){console.error("server stop error",c),i.push(c)}if(t==="database")try{await ft(o,e),n&&console.debug(`${t} state saved`)}catch(c){console.error("dump error",c),i.push(c)}try{await o?.close(),n&&console.debug(`${t} closed`)}catch(c){console.error("db close error",c),i.push(c)}if(i.length>0)throw new AggregateError(i,`Failed to close ${t} properly`)},connectionString:ht(s),port:s}}async function gt(t,e){let r=await e.readDatabaseDump();r&&(await t.exec(`${r}; set search_path to public;`),e.debug&&console.debug("successfully loaded database state from a dump file"))}async function ft(t,e){await e.writeDatabaseDump(()=>lt({args:[],pg:t}))}function ht(t){return`${mt}:${t}/${d.database}?${pt.toString()}`}import{copyFile as bt,mkdir as yt,writeFile as wt}from"fs/promises";import{join as L}from"pathe";import{lock as St}from"proper-lockfile";import{read as Pt}from"read-last-lines";import{process as vt}from"std-env";var j=Symbol("initialize"),h=class{databasePort;debug;dryRun;name;persistenceMode;port;shadowDatabasePort;constructor(e){this.databasePort=e.databasePort??51214,this.debug=e.debug??!1,this.dryRun=e.dryRun??!1,this.name=e.name??"default",this.persistenceMode=e.persistenceMode,this.port=e.port??51213,this.shadowDatabasePort=e.shadowDatabasePort??51215}static async get(e){let r=e?.dryRun!==!0&&e?.persistenceMode!=="stateless"?new F(e):new M(e);return await r[j](),r}},M=class extends h{constructor(e){super({...e,persistenceMode:"stateless"})}async[j](){}async close(){}readDatabaseDump(){return Promise.resolve(null)}async writeDatabaseDump(){}async writeServerDump(){}},F=class extends h{#r;#e;#t;#s;#o=null;constructor(e){super({...e,persistenceMode:"stateful"}),this.#r=J(this.name),this.#e=L(this.#r,"ppg.sql"),this.#t=`${this.#e}.bak`,this.#s=L(this.#r,"server.json")}async[j](){await yt(this.#r,{recursive:!0}),this.debug&&console.debug(`using data directory: ${this.#r}`);try{this.#o=await St(this.#r,{lockfilePath:L(this.#r,".lock")}),this.debug&&console.debug(`obtained lock on: ${this.#r}`),await this.writeServerDump()}catch(e){throw e instanceof Error&&"code"in e&&e.code==="ELOCKED"?new Error(`A server with the name "${this.name}" is already running. `):e}}async close(){this.#o!=null&&(await this.#o(),this.debug&&console.debug(`released lock on: ${this.#r}`))}async readDatabaseDump(){let[e,r]=await Promise.all([$(this.#e),$(this.#t)]),n=Number.isFinite(e)?this.#e:null,o=Number.isFinite(r)?this.#t:null,s=(e>=r?[n,o]:[o,n]).filter(Boolean);for(let a of s){if(!await this.#a(a)){this.debug&&console.debug(`database dump not completed at: ${a}`);continue}let i=await Y(a);if(!i){this.debug&&console.debug(`database dump no longer available at: ${a}`);continue}if(!this.#n(i)){this.debug&&console.debug(`database dump no longer completed at: ${a}`);continue}return this.debug&&console.debug(`using database dump from: ${a}`),i}return this.debug&&console.debug("no completed database dumps found"),null}async writeDatabaseDump(e){let r=!1;await this.#a(this.#e)&&(await bt(this.#e,this.#t),r=!0,this.debug&&console.debug(`backed up dump to: ${this.#t}`));let n=await e();if(await W(n,this.#e),this.debug&&console.debug(`dumped database to: ${this.#e}`),!r)return;let o=await X(this.#t);this.debug&&o&&console.debug(`deleted backup dump: ${this.#t}`)}async writeServerDump(e){await wt(this.#s,`${JSON.stringify({name:this.name,version:"1",pid:vt.pid,port:this.port,databasePort:this.databasePort,shadowDatabasePort:this.shadowDatabasePort,exports:e},null,2)}
8
+ `,{encoding:"utf-8"})}async#a(e){if(!await D(e))return!1;let r=await Pt(e,5,"utf-8");return this.debug&&console.debug(`last lines of checked dump: ${r}`),this.#n(r)}#n(e){return e.includes("-- PostgreSQL database dump complete")}};async function Ir(t){let e=await h.get(t),[r,n]=await Promise.all([C("database",e),C("shadow_database",e)]),o=await ge(r,e),s=`prisma+postgres://localhost:${o.port}/?${new URLSearchParams({api_key:V({databaseUrl:r.connectionString,shadowDatabaseUrl:n.connectionString})}).toString()}`,a={accelerate:{url:o.url},database:{connectionString:r.connectionString},ppg:{url:s},shadowDatabase:{connectionString:n.connectionString}};return await e.writeServerDump(a),{...a,close:()=>i(e,[o,r,n])};async function i(c,l){let m=(await Promise.allSettled(l.map(u=>u.close()))).filter(u=>u.status==="rejected").map(u=>new Error(u.reason));try{await c.close()}catch(u){m.push(u)}if(m.length>0)throw new AggregateError(m,"Failed to close some servers")}}export{Z as DEFAULT_DATABASE_PORT,ee as DEFAULT_SERVER_PORT,te as DEFAULT_SHADOW_DATABASE_PORT,f as PortNotAvailableError,Ir as unstable_startServer};
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@prisma/dev",
3
- "version": "0.0.0-dev.202505221232",
3
+ "version": "0.0.0-dev.202505221432",
4
4
  "description": "A local Prisma Postgres server for development and testing",
5
5
  "type": "module",
6
6
  "main": "./dist/index.js",
@@ -47,6 +47,7 @@
47
47
  "pako": "2.1.0",
48
48
  "pathe": "2.0.3",
49
49
  "proper-lockfile": "4.1.2",
50
+ "read-last-lines": "1.8.0",
50
51
  "std-env": "3.8.1",
51
52
  "valibot": "1.1.0"
52
53
  },