@prisma/dev 0.0.0-dev.202505231310 → 0.0.0-dev.202505231321

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 +1 -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 tt}from"http";import{promisify as rt}from"util";import{serve as nt}from"@hono/node-server";import ot from"@prisma/get-platform";import{logger as st}from"hono/logger";import{Hono as pe}from"hono/tiny";import{validator as S}from"hono/validator";import{process as at}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 $,url as N}from"valibot";var q=/^(postgres|postgresql):\/\//,Se=x($(),ye(),he({databaseUrl:x($(),N(),U(q)),shadowDatabaseUrl:be(x($(),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 $e=Ae({tags:xe([Te(ve(Oe()),De(1)),Ee("all")])});async function K(t){let{output:e,success:r}=Re($e,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{proxySignals as Ge}from"foreground-child/proxy-signals";import{process as ze}from"std-env";import{createWriteStream as Ie,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(Ie(e,{encoding:"utf-8"})))}async function X(t){try{return await Me(t),!0}catch{return!1}}async function I(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((s,a)=>{t=s,e=a}),n=s=>{n=o=null,e(s)},o=s=>{o=n=null,t(s)};return{fail:s=>n?.(s),promise:r,succeed:s=>o?.(s)}}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{fail:e,promise:r,succeed:n}=A();this.#t=r;let o=await this.#s();this.#e.debug&&console.debug("spinning up engine at path...",o);let s=Ne(o,["--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});Ge(s),s.stderr.setEncoding("utf8"),s.stdout.setEncoding("utf8");let a=i=>{let c=i.split(`
3
- `).find(u=>u.includes("Started query engine http server"));if(!c)return;s.stdout.removeListener("data",a);let{fields:l}=JSON.parse(c);if(l==null)return e(new Error(`Unexpected data during initialization, "fields" are missing: ${i}`));let{ip:b,port:m}=l;if(b==null||m==null)return e(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}`));n({childProcess:s,url:`http://${b}:${m}`})};s.once("error",i=>{this.#t=null,e(new w(String(i)))}),s.once("exit",(i,c)=>{this.#t=null,e(new w(`Query Engine exited with code ${i} and signal ${c}`))}),s.stdout.on("data",a),s.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),(ze.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 Je=/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 {
2
+ import{createServer as nt}from"http";import{promisify as ot}from"util";import{serve as st}from"@hono/node-server";import at from"@prisma/get-platform";import{logger as it}from"hono/logger";import{Hono as ge}from"hono/tiny";import{validator as S}from"hono/validator";import{process as ct}from"std-env";import{HTTPException as E}from"hono/http-exception";import{object as ye,optional as we,parseJson as Se,pipe as O,regex as N,safeParse as Pe,string as $,url as q}from"valibot";var V=/^(postgres|postgresql):\/\//,ve=O($(),Se(),ye({databaseUrl:O($(),q(),N(V)),shadowDatabaseUrl:we(O($(),q(),N(V)))}));function K(t){return Buffer.from(JSON.stringify(t),"utf8").toString("base64url")}function Ee(t){let e=Buffer.from(t,"base64url").toString("utf8"),{issues:r,output:n,success:o}=Pe(ve,e,{abortEarly:!0});return o?[null,n]:[r]}var g=(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]=Ee(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 De,literal as Ae,minLength as Te,object as Re,pipe as xe,safeParse as Oe,string as $e,union as ke}from"valibot";var Ie=Re({tags:ke([xe(De($e()),Te(1)),Ae("all")])});async function G(t){let{output:e,success:r}=Oe(Ie,await t.req.json(),{abortEarly:!0});return r?e:t.text("Invalid input",400)}import{spawn as Ve}from"child_process";import{once as Ke}from"events";import{mkdir as Ge}from"fs/promises";import{join as ze}from"path";import{proxySignals as Je}from"foreground-child/proxy-signals";import{process as Qe}from"std-env";import{createWriteStream as He,WriteStream as _e}from"fs";import{access as Be,chmod as Le,constants as Ce,readFile as Me,stat as Fe,unlink as je,writeFile as Ue}from"fs/promises";import Ne from"env-paths";import{inflate as qe}from"pako";var z=Ne("prisma-dev");function J(t,e){return`${z.cache}/engine/${t}/${e}`}function Q(t){return`${z.data}/${t}`}async function D(t){try{return await Be(t,Ce.F_OK),!0}catch(e){if(I(e))return!1;throw e}}async function W(t,e){let r=qe(t);await Ue(e,r),await Le(e,"755")}async function X(t,e){await t.stream().pipeTo(_e.toWeb(He(e,{encoding:"utf-8"})))}async function Y(t){try{return await je(t),!0}catch{return!1}}async function k(t){try{return(await Fe(t)).mtimeMs}catch(e){if(I(e))return-1/0;throw e}}function I(t){return t!=null&&typeof t=="object"&&"code"in t&&t.code==="ENOENT"}async function Z(t){try{return await Me(t,{encoding:"utf-8"})}catch(e){if(I(e))return null;throw e}}function A(){let t,e,r=new Promise((s,a)=>{t=s,e=a}),n=s=>{n=o=null,e(s)},o=s=>{o=n=null,t(s)};return{fail:s=>n?.(s),promise:r,succeed:s=>o?.(s)}}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 f.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 f.fromResponse(s);return await s.json()}async start(){if(this.#t!=null)return;let{fail:e,promise:r,succeed:n}=A();this.#t=r;let o=await this.#s();this.#e.debug&&console.debug("spinning up engine at path...",o);let s=Ve(o,["--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});Je(s),s.stderr.setEncoding("utf8"),s.stdout.setEncoding("utf8");let a=u=>{let m=u.split(`
3
+ `).find(be=>be.includes("Started query engine http server"));if(!m)return;s.stdout.removeListener("data",a);let{fields:p}=JSON.parse(m);if(p==null)return e(new Error(`Unexpected data during initialization, "fields" are missing: ${u}`));let{ip:l,port:U}=p;if(l==null||U==null)return e(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: ${u}`));n({childProcess:s,url:`http://${l}:${U}`})},i=u=>{this.#t=null,e(new w(String(u))),s.removeListener("exit",c),s.kill()};s.once("error",i);let c=(u,m)=>{this.#t=null,e(new w(`Query Engine exited with code ${u} and signal ${m}`))};s.once("exit",c),s.stdout.on("data",a),s.stderr.on("data",console.error),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 Ke(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=J(this.#e.clientVersion,e);this.#e.debug&&console.debug("cache directory path",r),await Ge(r,{recursive:!0});let{platform:n}=this.#e.platform,o=n==="windows"?".exe":"",s=ze(r,`query-engine-${n}${o}`);return this.#e.debug&&console.debug("engine binary path",s),(Qe.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 W(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 f.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 f?e.text(t.responseBody,t.statusCode):e.body(null,500)}var w=class extends Error{name="EngineStartError"},f=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 ee=51214,te=51213,re=51215,h=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 ne}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 oe(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 We=/datasource\s+db\s+\{\s*provider\s*=\s*"postgres(!?ql)?"\s+url\s*=\s*.+\s*\}/;async function se(t,e){let r=ne.from(t,"base64").toString("utf8"),n=`datasource db {
5
5
  provider = "postgresql"
6
6
  url = "${e}"
7
- }`,o=r.replace(Je,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 Qe,minValue as ae,number as _,object as We,optional as Xe,pipe as ie,safeParse as ce,string as de,union as Ye}from"valibot";var Ze=We({isolation_level:Xe(de()),max_wait:ie(_(),se(),ae(0)),timeout:ie(_(),se(),ae(0))});async function ue(t){let{issues:e,output:r,success:n}=ce(Ze,await t.req.json(),{abortEarly:!0});return n?r:t.json({EngineNotStarted:{reason:"InvalidRequest",issues:e}},400)}var et=Qe({id:Ye([de(),_()])});function le(t,e){let{output:r,success:n}=ce(et,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 it(r,t,e),{fail:o,promise:s,succeed:a}=A(),i=nt({createServer:tt,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([rt(e.close.bind(e))(),y.stopAll()])},port:t,url:`http://localhost:${t}`}}async function it(t,e,r){let{debug:n}=r,o=new pe,s=await ot.getPlatformInfo();return n&&console.debug("platform info: %s",JSON.stringify(s)),n&&o.use("*",st()),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 ct="/:clientVersion/:schemaHash",v=P.basePath(ct);P.route("/",v);var dt=["/graphql","/itx/:transactionId/graphql"];v.on("POST",[...dt],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:at.env.PRISMA_DEV_FORCE_CLIENT_VERSION||e.param("clientVersion"),debug:t.get("debug"),platform:t.get("platform"),schemaHash:o})}import{PGlite as ut}from"@electric-sql/pglite";import{PGLiteSocketServer as lt}from"@electric-sql/pglite-socket";import{pgDump as mt}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"},pt=`postgres://${d.username}:${d.password}@localhost`,gt=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 ut.create({database:d.database,dataDir:"memory://",debug:n?5:void 0,defaultDataTransferContainer:"file",username:d.username});await o.waitReady,t==="database"&&await ft(o,e);let s=new lt({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 ${s}`),{...d,close:async()=>{let i=[];try{await a?.stop(),n&&console.debug(`${t} server stopped on port ${s}`)}catch(c){console.error("server stop error",c),i.push(c)}if(t==="database")try{await ht(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:bt(s),port:s}}async function ft(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 ht(t,e){await e.writeDatabaseDump(()=>mt({args:[],pg:t}))}function bt(t){return`${pt}:${t}/${d.database}?${gt.toString()}`}import{copyFile as yt,mkdir as wt,writeFile as St}from"fs/promises";import{join as L}from"pathe";import{lock as Pt}from"proper-lockfile";import{readLastLines as vt}from"read-last-lines-ts";import{process as Et}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 wt(this.#r,{recursive:!0}),this.debug&&console.debug(`using data directory: ${this.#r}`);try{this.#o=await Pt(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([I(this.#e),I(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 yt(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 St(this.#s,`${JSON.stringify({name:this.name,version:"1",pid:Et.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=vt(e,5).toString("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};
7
+ }`,o=r.replace(We,n),s=await H(o);return{base64Override:ne.from(o,"utf8").toString("base64"),overrideHash:s}}function x(t){let{req:e}=t;return{traceparent:e.header("traceparent"),"X-capture-telemetry":e.header("X-capture-telemetry")}}import{integer as ae,looseObject as Xe,minValue as ie,number as _,object as Ye,optional as Ze,pipe as ce,safeParse as de,string as ue,union as et}from"valibot";var tt=Ye({isolation_level:Ze(ue()),max_wait:ce(_(),ae(),ie(0)),timeout:ce(_(),ae(),ie(0))});async function le(t){let{issues:e,output:r,success:n}=de(tt,await t.req.json(),{abortEarly:!0});return n?r:t.json({EngineNotStarted:{reason:"InvalidRequest",issues:e}},400)}var rt=Xe({id:et([ue(),_()])});function me(t,e){let{output:r,success:n}=de(rt,t);return n?r:e.json({EngineMalfunction:{}},500)}async function fe(t,e){let{port:r}=e;if(e.dryRun)return pe(r,null);let n=await dt(r,t,e),{fail:o,promise:s,succeed:a}=A(),i=st({createServer:nt,fetch:n.fetch,port:r},()=>{a()});return i.on("error",c=>{if(typeof c=="object"&&"code"in c&&c.code==="EADDRINUSE")return o(new h(r,"server"));console.error(c)}),await s,pe(r,i)}function pe(t,e){return{async close(){e&&await Promise.allSettled([ot(e.close.bind(e))(),y.stopAll()])},port:t,url:`http://localhost:${t}`}}async function dt(t,e,r){let{debug:n}=r,o=new ge,s=await at.getPlatformInfo();return n&&console.debug("platform info: %s",JSON.stringify(s)),n&&o.use("*",it()),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 ge;P.get("/health",t=>t.json({name:t.get("serverState").name}));P.post("/invalidate",S("header",g),async t=>{let e=await G(t);return e instanceof Response?e:t.body(null)});var ut="/:clientVersion/:schemaHash",v=P.basePath(ut);P.route("/",v);var lt=["/graphql","/itx/:transactionId/graphql"];v.on("POST",[...lt],S("header",g),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,{...x(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",g),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,x(t));return t.json(a)}catch(r){return T(r,t)}});v.put("/schema",S("header",g),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 se(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",g),async t=>{let{req:e}=t,r=await le(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,x(t)),s=me(o,t);if(s instanceof Response)return s;let{id:a}=s,i=e.param("clientVersion"),c=t.get("port"),u=t.get("protocol"),m=e.param("schemaHash");return t.json({...o,"data-proxy":{endpoint:`${u}://localhost:${c}/${i}/${m}/itx/${a}`}})}catch(n){return T(n,t)}});async function B(t){let{req:e}=t,r=oe(t);if(r instanceof Response)return r;let{base64Override:n,overrideHash:o}=r.schemas;return await y.get({base64Schema:n,clientVersion:ct.env.PRISMA_DEV_FORCE_CLIENT_VERSION||e.param("clientVersion"),debug:t.get("debug"),platform:t.get("platform"),schemaHash:o})}import{PGlite as mt}from"@electric-sql/pglite";import{PGLiteSocketServer as pt}from"@electric-sql/pglite-socket";import{pgDump as gt}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"},ft=`postgres://${d.username}:${d.password}@localhost`,ht=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 L(t,e){let r=t==="database"?e.databasePort:e.shadowDatabasePort;if(e.dryRun)return he(t,e,{db:null,port:r,server:null});let{debug:n}=e,o=await mt.create({database:d.database,dataDir:"memory://",debug:n?5:void 0,defaultDataTransferContainer:"file",username:d.username});await o.waitReady,t==="database"&&await bt(o,e);let s=new pt({db:o,inspect:n,port:r});try{await s.start()}catch(a){throw a instanceof Error&&"code"in a&&a.code==="EADDRINUSE"?new h(r,t):a}return he(t,e,{db:o,port:r,server:s})}function he(t,e,r){let{debug:n}=e,{db:o,port:s,server:a}=r||{};return n&&console.debug(`${t} server started on port ${s}`),{...d,close:async()=>{let i=[];try{await a?.stop(),n&&console.debug(`${t} server stopped on port ${s}`)}catch(c){console.error("server stop error",c),i.push(c)}if(t==="database")try{await yt(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:wt(s),port:s}}async function bt(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 yt(t,e){await e.writeDatabaseDump(()=>gt({args:[],pg:t}))}function wt(t){return`${ft}:${t}/${d.database}?${ht.toString()}`}import{copyFile as St,mkdir as Pt,writeFile as vt}from"fs/promises";import{join as C}from"pathe";import{lock as Et}from"proper-lockfile";import{readLastLines as Dt}from"read-last-lines-ts";import{process as At}from"std-env";var j=Symbol("initialize"),b=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 b{constructor(e){super({...e,persistenceMode:"stateless"})}async[j](){}async close(){}readDatabaseDump(){return Promise.resolve(null)}async writeDatabaseDump(){}async writeServerDump(){}},F=class extends b{#r;#e;#t;#s;#o=null;constructor(e){super({...e,persistenceMode:"stateful"}),this.#r=Q(this.name),this.#e=C(this.#r,"ppg.sql"),this.#t=`${this.#e}.bak`,this.#s=C(this.#r,"server.json")}async[j](){await Pt(this.#r,{recursive:!0}),this.debug&&console.debug(`using data directory: ${this.#r}`);try{this.#o=await Et(this.#r,{lockfilePath:C(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([k(this.#e),k(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 Z(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 St(this.#e,this.#t),r=!0,this.debug&&console.debug(`backed up dump to: ${this.#t}`));let n=await e();if(await X(n,this.#e),this.debug&&console.debug(`dumped database to: ${this.#e}`),!r)return;let o=await Y(this.#t);this.debug&&o&&console.debug(`deleted backup dump: ${this.#t}`)}async writeServerDump(e){await vt(this.#s,`${JSON.stringify({name:this.name,version:"1",pid:At.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=Dt(e,5).toString("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 Hr(t){let e=await b.get(t),[r,n]=await Promise.all([L("database",e),L("shadow_database",e)]),o=await fe(r,e),s=`prisma+postgres://localhost:${o.port}/?${new URLSearchParams({api_key:K({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,u){let p=(await Promise.allSettled(u.map(l=>l.close()))).filter(l=>l.status==="rejected").map(l=>new Error(l.reason));try{await c.close()}catch(l){p.push(l)}if(p.length>0)throw new AggregateError(p,"Failed to close some servers")}}export{ee as DEFAULT_DATABASE_PORT,te as DEFAULT_SERVER_PORT,re as DEFAULT_SHADOW_DATABASE_PORT,h as PortNotAvailableError,Hr as unstable_startServer};
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@prisma/dev",
3
- "version": "0.0.0-dev.202505231310",
3
+ "version": "0.0.0-dev.202505231321",
4
4
  "description": "A local Prisma Postgres server for development and testing",
5
5
  "type": "module",
6
6
  "main": "./dist/index.js",