@prisma/dev 0.0.0-dev.202505192354 → 0.0.0-dev.202505200036

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.d.ts CHANGED
@@ -32,6 +32,12 @@ interface StartServerOptions {
32
32
  * Defaults to `false`.
33
33
  */
34
34
  debug?: boolean;
35
+ /**
36
+ * Whether to run the server in dry run mode.
37
+ *
38
+ * Defaults to `false`.
39
+ */
40
+ dryRun?: boolean;
35
41
  /**
36
42
  * The port the shadow database server will listen on.
37
43
  *
package/dist/index.js CHANGED
@@ -1,7 +1,7 @@
1
1
  import { createRequire } from 'module'; const require = createRequire(import.meta.url);
2
- import{createServer as ke}from"http";import{promisify as Be}from"util";import{serve as Le}from"@hono/node-server";import je from"@prisma/get-platform";import{logger as Ue}from"hono/logger";import{Hono as te}from"hono/tiny";import{validator as w}from"hono/validator";import{HTTPException as v}from"hono/http-exception";import{object as re,optional as oe,parseJson as se,pipe as x,regex as $,safeParse as ae,string as H,url as k}from"valibot";var B=/^(postgres|postgresql):\/\//,ie=x(H(),se(),re({databaseUrl:x(H(),k(),$(B)),shadowDatabaseUrl:oe(x(H(),k(),$(B)))}));function L(e){return Buffer.from(JSON.stringify(e),"utf8").toString("base64url")}function ce(e){let t=Buffer.from(e,"base64url").toString("utf8"),{issues:n,output:r,success:o}=ae(ie,t,{abortEarly:!0});return o?[null,r]:[n]}var m=(e,t)=>{let{authorization:n}=e;if(!n)throw new v(401,{message:"Missing API Key"});let[r,o="",s]=n.split(" ");if(r!=="Bearer"||s)throw new v(401,{message:"Invalid API Key"});let[a,i]=ce(o);if(a)throw new v(401,{message:"Invalid API Key",cause:a.join(", ")});let{databaseUrl:c}=i;if(c!==t.var.db.connectionString)throw new v(401,{message:"Unauthorized"});return{decodedAPIKey:i}};import{array as le,literal as de,minLength as ue,object as me,pipe as pe,safeParse as ge,string as fe,union as he}from"valibot";var ye=me({tags:he([pe(le(fe()),ue(1)),de("all")])});async function j(e){let{output:t,success:n}=ge(ye,await e.req.json(),{abortEarly:!0});return n?t:e.text("Invalid input",400)}import{spawn as Ae}from"child_process";import{once as Te}from"events";import{mkdir as Re}from"fs/promises";import{join as Oe}from"path";import{chmod as we,stat as Se,writeFile as be}from"fs/promises";import Pe from"env-paths";import{inflate as ve}from"pako";var Ee=Pe("prisma-dev").cache;function U(e,t){return`${Ee}/${e}/${t}`}async function q(e){try{return await Se(e),!0}catch(t){if(t!=null&&typeof t=="object"&&"code"in t&&t.code==="ENOENT")return!1;throw t}}async function F(e,t){let n=ve(e);await be(t,n),await we(t,"755")}function E(){let e,t,n=new Promise((r,o)=>{e=r,t=o});return{fail:t,promise:n,succeed:e}}var h=class e{static#n=new Map;#e;#t;constructor(t){this.#e=t,this.#t=null}static async get(t){let n=`${t.schemaHash}:${t.clientVersion}`;try{let r=e.#n.get(n);if(r)return r;let o=new e(t);return e.#n.set(n,o),t.debug&&console.debug("starting engine...",t),await o.start(),t.debug&&console.debug("engine started!"),o}finally{e.stopAll(n)}}static async stopAll(t){let r=(await Promise.allSettled(Array.from(e.#n.entries()).filter(([o])=>o!==t).map(async([o,s])=>{try{await s.stop()}finally{e.#n.delete(o)}}))).filter(o=>o.status==="rejected").map(o=>o.reason);if(r.length>0)throw new AggregateError(r,"Failed to stop engines")}async commitTransaction(t,n){return await this.#o(t,n,"commit")}async request(t,n){let{url:r}=await this.#t,o=this.#r(n),s=await fetch(r,{body:typeof t=="string"?t:JSON.stringify(t),headers:{...o,"Content-Type":"application/json"},method:"POST"});if(!s.ok)throw await p.fromResponse(s);return await s.text()}async rollbackTransaction(t,n){return await this.#o(t,n,"rollback")}async startTransaction(t,n){let{url:r}=await this.#t,o=this.#r(n),s=await fetch(`${r}/transaction/start`,{body:JSON.stringify(t),headers:{...o,"Content-Type":"application/json"},method:"POST"});if(!s.ok)throw await p.fromResponse(s);return await s.json()}async start(){if(this.#t!=null)return;let t=await this.#s();this.#e.debug&&console.debug("spinning up engine at path...",t);let n=Ae(t,["--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});n.stderr.setEncoding("utf8"),n.stdout.setEncoding("utf8");let{fail:r,promise:o,succeed:s}=E();this.#t=o;let a=i=>{let c=i.split(`
3
- `).find(P=>P.includes("Started query engine http server"));if(!c)return;n.stdout.removeListener("data",a);let{fields:d}=JSON.parse(c);if(d==null)return r(new Error(`Unexpected data during initialization, "fields" are missing: ${i}`));let{ip:u,port:O}=d;if(u==null||O==null)return r(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:n,url:`http://${u}:${O}`})};n.once("error",i=>{this.#t=null,r(new y(String(i)))}),n.once("exit",(i,c)=>{this.#t=null,r(new y(`Query Engine exited with code ${i} and signal ${c}`))}),n.stdout.on("data",a),n.stderr.on("data",i=>{console.error(i)}),await this.#t}async stop(){if(this.#t==null)return;let{childProcess:t}=await this.#t;t.exitCode==null&&t.signalCode==null&&(t.kill(),await Te(t,"exit"))}async#s(){this.#e.debug&&console.debug("getting engine commit hash...");let t=await this.#a();this.#e.debug&&console.debug("got engine commit hash",t);let n=U(this.#e.clientVersion,t);this.#e.debug&&console.debug("cache directory path",n),await Re(n,{recursive:!0});let{platform:r}=this.#e.platform,o=r==="windows"?".exe":"",s=Oe(n,`query-engine-${r}${o}`);return this.#e.debug&&console.debug("engine binary path",s),(process.env.PRISMA_DEV_FORCE_ENGINE_DOWNLOAD==="1"||await q(s)===!1)&&await this.#i({commitHash:t,extension:o,engineBinaryPath:s}),s}async#a(){let t=await fetch(`https://registry.npmjs.org/@prisma/client/${this.#e.clientVersion}`);if(!t.ok)throw new Error(`Couldn't fetch package.json from npm registry, status code: ${t.status}`);let r=(await t.json()).devDependencies?.["@prisma/engines-version"];if(!r)throw new Error("Couldn't find engines version in package.json");let o=r.split(".").at(-1);if(!o)throw new Error("Couldn't find commit hash in engines version");return o}async#i(t){let{commitHash:n,extension:r,engineBinaryPath:o}=t,{binaryTarget:s}=this.#e.platform,a=`https://binaries.prisma.sh/all_commits/${n}/${s}/query-engine${r}.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 F(await i.arrayBuffer(),o),this.#e.debug&&console.debug("engine downloaded and saved at",o)}#r(t){let n={};for(let[r,o]of Object.entries(t))o!=null&&(n[r]=o);return n}async#o(t,n,r){let{url:o}=await this.#t,s=this.#r(n),a=await fetch(`${o}/transaction/${t}/${r}`,{headers:{...s,"Content-Type":"application/json"},method:"POST"});if(!a.ok)throw await p.fromResponse(a);try{return await a.json()}catch{return{}}}};function A(e,t){return console.error(e),e instanceof y?t.json({EngineNotStarted:{reason:{EngineStartupError:{logs:[],msg:e.message}}}},500):e instanceof p?t.text(e.responseBody,e.statusCode):t.body(null,500)}var y=class extends Error{name="EngineStartError"},p=class e extends Error{constructor(n,r,o){super(`${n}: Query Engine response status ${r}, body: ${o}`);this.action=n;this.statusCode=r;this.responseBody=o}name="EngineHttpError";static async fromResponse(n){let r=new URL(n.url),o=await n.text();return new e(r.pathname,n.status,o)}};var M=51213,N=51214,V=51215,g=class extends Error{constructor(n,r){super(`Port number \`${n}\` is not available for service ${r}.`);this.port=n;this.service=r}name="PortNotAvailableError"};import{Buffer as K}from"buffer";var T=new Map;async function I(e){let n=new TextEncoder().encode(e),r=await crypto.subtle.digest("SHA-256",n);return Array.from(new Uint8Array(r)).map(a=>a.toString(16).padStart(2,"0")).join("")}function z(e){let t=e.req.param("schemaHash"),n=T.get(t);return n==null?e.json({EngineNotStarted:{reason:"SchemaMissing"}},404):{schemaHash:t,schemas:n}}var xe=/datasource\s+db\s+\{\s*provider\s*=\s*"postgres(!?ql)?"\s+url\s*=\s*.+\s*\}/;async function G(e,t){let n=K.from(e,"base64").toString("utf8"),r=`datasource db {
2
+ import{createServer as je}from"http";import{promisify as Ue}from"util";import{serve as qe}from"@hono/node-server";import Fe from"@prisma/get-platform";import{logger as Me}from"hono/logger";import{Hono as re}from"hono/tiny";import{validator as S}from"hono/validator";import{HTTPException as P}from"hono/http-exception";import{object as ae,optional as ie,parseJson as ce,pipe as O,regex as k,safeParse as le,string as x,url as B}from"valibot";var L=/^(postgres|postgresql):\/\//,ue=O(x(),ce(),ae({databaseUrl:O(x(),B(),k(L)),shadowDatabaseUrl:ie(O(x(),B(),k(L)))}));function j(t){return Buffer.from(JSON.stringify(t),"utf8").toString("base64url")}function pe(t){let e=Buffer.from(t,"base64url").toString("utf8"),{issues:n,output:r,success:o}=le(ue,e,{abortEarly:!0});return o?[null,r]:[n]}var d=(t,e)=>{let{authorization:n}=t;if(!n)throw new P(401,{message:"Missing API Key"});let[r,o="",s]=n.split(" ");if(r!=="Bearer"||s)throw new P(401,{message:"Invalid API Key"});let[a,i]=pe(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 de,literal as me,minLength as ge,object as fe,pipe as he,safeParse as ye,string as Se,union as we}from"valibot";var be=fe({tags:we([he(de(Se()),ge(1)),me("all")])});async function U(t){let{output:e,success:n}=ye(be,await t.req.json(),{abortEarly:!0});return n?e:t.text("Invalid input",400)}import{spawn as Oe}from"child_process";import{once as xe}from"events";import{mkdir as He}from"fs/promises";import{join as Ie}from"path";import{chmod as ve,stat as Pe,writeFile as Ee}from"fs/promises";import Ae from"env-paths";import{inflate as Te}from"pako";var Re=Ae("prisma-dev").cache;function q(t,e){return`${Re}/${t}/${e}`}async function F(t){try{return await Pe(t),!0}catch(e){if(e!=null&&typeof e=="object"&&"code"in e&&e.code==="ENOENT")return!1;throw e}}async function M(t,e){let n=Te(t);await Ee(e,n),await ve(e,"755")}function E(){let t,e,n=new Promise((r,o)=>{t=r,e=o});return{fail:e,promise:n,succeed:t}}var h=class t{static#n=new Map;#e;#t;constructor(e){this.#e=e,this.#t=null}static async get(e){let n=`${e.schemaHash}:${e.clientVersion}`;try{let r=t.#n.get(n);if(r)return r;let o=new t(e);return t.#n.set(n,o),e.debug&&console.debug("starting engine...",e),await o.start(),e.debug&&console.debug("engine started!"),o}finally{t.stopAll(n)}}static async stopAll(e){let r=(await Promise.allSettled(Array.from(t.#n.entries()).filter(([o])=>o!==e).map(async([o,s])=>{try{await s.stop()}finally{t.#n.delete(o)}}))).filter(o=>o.status==="rejected").map(o=>o.reason);if(r.length>0)throw new AggregateError(r,"Failed to stop engines")}async commitTransaction(e,n){return await this.#o(e,n,"commit")}async request(e,n){let{url:r}=await this.#t,o=this.#r(n),s=await fetch(r,{body:typeof e=="string"?e:JSON.stringify(e),headers:{...o,"Content-Type":"application/json"},method:"POST"});if(!s.ok)throw await m.fromResponse(s);return await s.text()}async rollbackTransaction(e,n){return await this.#o(e,n,"rollback")}async startTransaction(e,n){let{url:r}=await this.#t,o=this.#r(n),s=await fetch(`${r}/transaction/start`,{body:JSON.stringify(e),headers:{...o,"Content-Type":"application/json"},method:"POST"});if(!s.ok)throw await m.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 n=Oe(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});n.stderr.setEncoding("utf8"),n.stdout.setEncoding("utf8");let{fail:r,promise:o,succeed:s}=E();this.#t=o;let a=i=>{let c=i.split(`
3
+ `).find(C=>C.includes("Started query engine http server"));if(!c)return;n.stdout.removeListener("data",a);let{fields:p}=JSON.parse(c);if(p==null)return r(new Error(`Unexpected data during initialization, "fields" are missing: ${i}`));let{ip:u,port:v}=p;if(u==null||v==null)return r(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:n,url:`http://${u}:${v}`})};n.once("error",i=>{this.#t=null,r(new y(String(i)))}),n.once("exit",(i,c)=>{this.#t=null,r(new y(`Query Engine exited with code ${i} and signal ${c}`))}),n.stdout.on("data",a),n.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 xe(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 n=q(this.#e.clientVersion,e);this.#e.debug&&console.debug("cache directory path",n),await He(n,{recursive:!0});let{platform:r}=this.#e.platform,o=r==="windows"?".exe":"",s=Ie(n,`query-engine-${r}${o}`);return this.#e.debug&&console.debug("engine binary path",s),(process.env.PRISMA_DEV_FORCE_ENGINE_DOWNLOAD==="1"||await F(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 r=(await e.json()).devDependencies?.["@prisma/engines-version"];if(!r)throw new Error("Couldn't find engines version in package.json");let o=r.split(".").at(-1);if(!o)throw new Error("Couldn't find commit hash in engines version");return o}async#i(e){let{commitHash:n,extension:r,engineBinaryPath:o}=e,{binaryTarget:s}=this.#e.platform,a=`https://binaries.prisma.sh/all_commits/${n}/${s}/query-engine${r}.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 M(await i.arrayBuffer(),o),this.#e.debug&&console.debug("engine downloaded and saved at",o)}#r(e){let n={};for(let[r,o]of Object.entries(e))o!=null&&(n[r]=o);return n}async#o(e,n,r){let{url:o}=await this.#t,s=this.#r(n),a=await fetch(`${o}/transaction/${e}/${r}`,{headers:{...s,"Content-Type":"application/json"},method:"POST"});if(!a.ok)throw await m.fromResponse(a);try{return await a.json()}catch{return{}}}};function A(t,e){return console.error(t),t instanceof y?e.json({EngineNotStarted:{reason:{EngineStartupError:{logs:[],msg:t.message}}}},500):t instanceof m?e.text(t.responseBody,t.statusCode):e.body(null,500)}var y=class extends Error{name="EngineStartError"},m=class t extends Error{constructor(n,r,o){super(`${n}: Query Engine response status ${r}, body: ${o}`);this.action=n;this.statusCode=r;this.responseBody=o}name="EngineHttpError";static async fromResponse(n){let r=new URL(n.url),o=await n.text();return new t(r.pathname,n.status,o)}};var N=51213,V=51214,K=51215,g=class extends Error{constructor(n,r){super(`Port number \`${n}\` is not available for service ${r}.`);this.port=n;this.service=r}name="PortNotAvailableError"};import{Buffer as G}from"buffer";var T=new Map;async function H(t){let n=new TextEncoder().encode(t),r=await crypto.subtle.digest("SHA-256",n);return Array.from(new Uint8Array(r)).map(a=>a.toString(16).padStart(2,"0")).join("")}function z(t){let e=t.req.param("schemaHash"),n=T.get(e);return n==null?t.json({EngineNotStarted:{reason:"SchemaMissing"}},404):{schemaHash:e,schemas:n}}var De=/datasource\s+db\s+\{\s*provider\s*=\s*"postgres(!?ql)?"\s+url\s*=\s*.+\s*\}/;async function J(t,e){let n=G.from(t,"base64").toString("utf8"),r=`datasource db {
5
5
  provider = "postgresql"
6
- url = "${t}"
7
- }`,o=n.replace(xe,r),s=await I(o);return{base64Override:K.from(o,"utf8").toString("base64"),overrideHash:s}}function R(e){let{req:t}=e;return{traceparent:t.header("traceparent"),"X-capture-telemetry":t.header("X-capture-telemetry")}}import{integer as J,looseObject as He,minValue as Q,number as C,object as Ie,optional as Ce,pipe as X,safeParse as W,string as Y,union as _e}from"valibot";var De=Ie({isolation_level:Ce(Y()),max_wait:X(C(),J(),Q(0)),timeout:X(C(),J(),Q(0))});async function Z(e){let{issues:t,output:n,success:r}=W(De,await e.req.json(),{abortEarly:!0});return r?n:e.json({EngineNotStarted:{reason:"InvalidRequest",issues:t}},400)}var $e=He({id:_e([Y(),C()])});function ee(e,t){let{output:n,success:r}=W($e,e);return r?n:t.json({EngineMalfunction:{}},500)}async function ne(e,t,n){let r=await qe(e,t,n),{fail:o,promise:s,succeed:a}=E(),i=Le({createServer:ke,fetch:r.fetch,port:e},()=>{a()});return i.on("error",c=>{if(typeof c=="object"&&"code"in c&&c.code==="EADDRINUSE")return o(new g(e,"accelerate"));console.error(c)}),await s,{async close(){await Promise.allSettled([Be(i.close.bind(i))(),h.stopAll()])},port:e,url:`http://localhost:${e}`}}async function qe(e,t,n){let r=new te,o=await je.getPlatformInfo();return n&&console.debug("platform info: %s",JSON.stringify(o)),n&&r.use("*",Ue()),r.use("*",async(s,a)=>(s.set("db",t),s.set("debug",!!n),s.set("platform",o),s.set("port",e),s.set("protocol","http"),await a())),r.route("/",S),r}var S=new te;S.get("/health",e=>e.text("ok"));S.post("/invalidate",w("header",m),async e=>{let t=await j(e);return t instanceof Response?t:e.body(null)});var Fe="/:clientVersion/:schemaHash",b=S.basePath(Fe);S.route("/",b);var Me=["/graphql","/itx/:transactionId/graphql"];b.on("POST",[...Me],w("header",m),async e=>{let{req:t}=e;try{let n=await _(e);if(n instanceof Response)return n;let r=await t.text(),o=t.param("transactionId"),s=await n.request(r,{...R(e),"X-transaction-id":o});return e.text(s)}catch(n){return A(n,e)}});b.basePath("/itx/:transactionId").on("POST",["/commit","/rollback"],w("header",m),async e=>{let{req:t}=e;try{let n=await _(e);if(n instanceof Response)return n;let o=`${t.routePath.split("/").filter(Boolean).at(-1)}Transaction`,s=t.param("transactionId"),a=await n[o](s,R(e));return e.json(a)}catch(n){return A(n,e)}});b.put("/schema",w("header",m),async e=>{let{req:t}=e,n=await t.text();if(!n)return e.text("Missing schema",400);let r=t.param("schemaHash"),o=T.get(r);if(o==null){if(r!==await I(n))return e.text("Schema hash mismatch",400);let s=await G(n,e.get("db").connectionString);return T.set(r,{base64Original:n,...s}),e.text(r)}return n!==o.base64Original?e.text("Schema mismatch",400):e.text(r)});b.post("/transaction/start",w("header",m),async e=>{let{req:t}=e,n=await Z(e);if(n instanceof Response)return n;try{let r=await _(e);if(r instanceof Response)return r;let o=await r.startTransaction(n,R(e)),s=ee(o,e);if(s instanceof Response)return s;let{id:a}=s,i=t.param("clientVersion"),c=e.get("port"),d=e.get("protocol"),u=t.param("schemaHash");return e.json({...o,"data-proxy":{endpoint:`${d}://localhost:${c}/${i}/${u}/itx/${a}`}})}catch(r){return A(r,e)}});async function _(e){let{req:t}=e,n=z(e);if(n instanceof Response)return n;let{base64Override:r,overrideHash:o}=n.schemas;return await h.get({base64Schema:r,clientVersion:process.env.PRISMA_DEV_FORCE_CLIENT_VERSION||t.param("clientVersion"),debug:e.get("debug"),platform:e.get("platform"),schemaHash:o})}import{PGlite as Ne}from"@electric-sql/pglite";import{PGLiteSocketServer as Ve}from"@electric-sql/pglite-socket";var l={connectionLimit:1,connectTimeout:0,database:"postgres",maxIdleConnectionLifetime:0,password:"postgres",poolTimeout:0,socketTimeout:0,sslMode:"disable",username:"postgres"},Ke=`postgresql://${l.username}:${l.password}@localhost`,ze=new URLSearchParams({connection_limit:String(l.connectionLimit),connect_timeout:String(l.connectTimeout),max_idle_connection_lifetime:String(l.maxIdleConnectionLifetime),pool_timeout:String(l.poolTimeout),socket_timeout:String(l.socketTimeout),sslmode:l.sslMode}).toString();async function D(e,t,n){let r=await Ne.create({database:l.database,dataDir:"memory://",debug:n?5:void 0,defaultDataTransferContainer:"file",username:l.username});await r.waitReady;let o=new Ve({db:r,inspect:n,port:e});try{await o.start()}catch(s){throw s instanceof Error&&"code"in s&&s.code==="EADDRINUSE"?new g(e,t):s}return n&&console.debug(`${t} server started on port %i`,e),{...l,async close(){try{await o.stop()}finally{await r.close()}},connectionString:Ge(e),port:e}}function Ge(e){return`${Ke}:${e}/${l.database}?${ze}`}async function Mt(e){let{acceleratePort:t=51213,databasePort:n=51214,debug:r=!1,shadowDatabasePort:o=51215}=e||{},[s,a]=await Promise.all([D(n,"database",r),D(o,"shadow_database",r)]),i=await ne(t,s,r),c=`prisma+postgres://localhost:${i.port}/?${new URLSearchParams({api_key:L({databaseUrl:s.connectionString,shadowDatabaseUrl:a.connectionString})}).toString()}`;return{accelerate:{url:i.url},close:()=>d([i,s,a]),database:{connectionString:s.connectionString},ppg:{url:c},shadowDatabase:{connectionString:a.connectionString}};async function d(u){let P=(await Promise.allSettled(u.map(f=>f.close()))).filter(f=>f.status==="rejected").map(f=>new Error(f.reason));if(P.length>0)throw new AggregateError(P,"Failed to close some servers")}}export{M as DEFAULT_ACCELERATE_PORT,N as DEFAULT_DATABASE_PORT,V as DEFAULT_SHADOW_DATABASE_PORT,g as PortNotAvailableError,Mt as unstable_startServer};
6
+ url = "${e}"
7
+ }`,o=n.replace(De,r),s=await H(o);return{base64Override:G.from(o,"utf8").toString("base64"),overrideHash:s}}function R(t){let{req:e}=t;return{traceparent:e.header("traceparent"),"X-capture-telemetry":e.header("X-capture-telemetry")}}import{integer as Q,looseObject as _e,minValue as X,number as I,object as Ce,optional as $e,pipe as W,safeParse as Y,string as Z,union as ke}from"valibot";var Be=Ce({isolation_level:$e(Z()),max_wait:W(I(),Q(),X(0)),timeout:W(I(),Q(),X(0))});async function ee(t){let{issues:e,output:n,success:r}=Y(Be,await t.req.json(),{abortEarly:!0});return r?n:t.json({EngineNotStarted:{reason:"InvalidRequest",issues:e}},400)}var Le=_e({id:ke([Z(),I()])});function te(t,e){let{output:n,success:r}=Y(Le,t);return r?n:e.json({EngineMalfunction:{}},500)}async function oe(t){let{dbServer:e,debug:n,dryRun:r,port:o}=t;if(r)return ne(o,null);let s=await Ne(o,e,n),{fail:a,promise:i,succeed:c}=E(),p=qe({createServer:je,fetch:s.fetch,port:o},()=>{c()});return p.on("error",u=>{if(typeof u=="object"&&"code"in u&&u.code==="EADDRINUSE")return a(new g(o,"accelerate"));console.error(u)}),await i,ne(o,p)}function ne(t,e){return{async close(){e&&await Promise.allSettled([Ue(e.close.bind(e))(),h.stopAll()])},port:t,url:`http://localhost:${t}`}}async function Ne(t,e,n){let r=new re,o=await Fe.getPlatformInfo();return n&&console.debug("platform info: %s",JSON.stringify(o)),n&&r.use("*",Me()),r.use("*",async(s,a)=>(s.set("db",e),s.set("debug",!!n),s.set("platform",o),s.set("port",t),s.set("protocol","http"),await a())),r.route("/",w),r}var w=new re;w.get("/health",t=>t.text("ok"));w.post("/invalidate",S("header",d),async t=>{let e=await U(t);return e instanceof Response?e:t.body(null)});var Ve="/:clientVersion/:schemaHash",b=w.basePath(Ve);w.route("/",b);var Ke=["/graphql","/itx/:transactionId/graphql"];b.on("POST",[...Ke],S("header",d),async t=>{let{req:e}=t;try{let n=await D(t);if(n instanceof Response)return n;let r=await e.text(),o=e.param("transactionId"),s=await n.request(r,{...R(t),"X-transaction-id":o});return t.text(s)}catch(n){return A(n,t)}});b.basePath("/itx/:transactionId").on("POST",["/commit","/rollback"],S("header",d),async t=>{let{req:e}=t;try{let n=await D(t);if(n instanceof Response)return n;let o=`${e.routePath.split("/").filter(Boolean).at(-1)}Transaction`,s=e.param("transactionId"),a=await n[o](s,R(t));return t.json(a)}catch(n){return A(n,t)}});b.put("/schema",S("header",d),async t=>{let{req:e}=t,n=await e.text();if(!n)return t.text("Missing schema",400);let r=e.param("schemaHash"),o=T.get(r);if(o==null){if(r!==await H(n))return t.text("Schema hash mismatch",400);let s=await J(n,t.get("db").connectionString);return T.set(r,{base64Original:n,...s}),t.text(r)}return n!==o.base64Original?t.text("Schema mismatch",400):t.text(r)});b.post("/transaction/start",S("header",d),async t=>{let{req:e}=t,n=await ee(t);if(n instanceof Response)return n;try{let r=await D(t);if(r instanceof Response)return r;let o=await r.startTransaction(n,R(t)),s=te(o,t);if(s instanceof Response)return s;let{id:a}=s,i=e.param("clientVersion"),c=t.get("port"),p=t.get("protocol"),u=e.param("schemaHash");return t.json({...o,"data-proxy":{endpoint:`${p}://localhost:${c}/${i}/${u}/itx/${a}`}})}catch(r){return A(r,t)}});async function D(t){let{req:e}=t,n=z(t);if(n instanceof Response)return n;let{base64Override:r,overrideHash:o}=n.schemas;return await h.get({base64Schema:r,clientVersion:process.env.PRISMA_DEV_FORCE_CLIENT_VERSION||e.param("clientVersion"),debug:t.get("debug"),platform:t.get("platform"),schemaHash:o})}import{PGlite as Ge}from"@electric-sql/pglite";import{PGLiteSocketServer as ze}from"@electric-sql/pglite-socket";var l={connectionLimit:1,connectTimeout:0,database:"postgres",maxIdleConnectionLifetime:0,password:"postgres",poolTimeout:0,socketTimeout:0,sslMode:"disable",username:"postgres"},Je=`postgresql://${l.username}:${l.password}@localhost`,Qe=new URLSearchParams({connection_limit:String(l.connectionLimit),connect_timeout:String(l.connectTimeout),max_idle_connection_lifetime:String(l.maxIdleConnectionLifetime),pool_timeout:String(l.poolTimeout),socket_timeout:String(l.socketTimeout),sslmode:l.sslMode});async function _(t){let{debug:e,dryRun:n,port:r,purpose:o}=t;if(n)return se(r,null,null);let s=await Ge.create({database:l.database,dataDir:"memory://",debug:e?5:void 0,defaultDataTransferContainer:"file",username:l.username});await s.waitReady;let a=new ze({db:s,inspect:e,port:r});try{await a.start()}catch(i){throw i instanceof Error&&"code"in i&&i.code==="EADDRINUSE"?new g(r,o):i}return e&&console.debug(`${o} server started on port %i`,r),se(r,s,a)}function se(t,e,n){return{...l,close:async()=>{if(!(!e||!n))try{await n.stop()}finally{await e.close()}},connectionString:Xe(t),port:t}}function Xe(t){return`${Je}:${t}/${l.database}?${Qe.toString()}`}async function Gt(t){let{acceleratePort:e=51213,databasePort:n=51214,debug:r=!1,dryRun:o=!1,shadowDatabasePort:s=51215}=t||{},[a,i]=await Promise.all([_({debug:r,dryRun:o,port:n,purpose:"database"}),_({debug:r,dryRun:o,port:s,purpose:"shadow_database"})]),c=await oe({dbServer:a,debug:r,dryRun:o,port:e}),p=`prisma+postgres://localhost:${c.port}/?${new URLSearchParams({api_key:j({databaseUrl:a.connectionString,shadowDatabaseUrl:i.connectionString})}).toString()}`;return{accelerate:{url:c.url},close:()=>u([c,a,i]),database:{connectionString:a.connectionString},ppg:{url:p},shadowDatabase:{connectionString:i.connectionString}};async function u(v){let $=(await Promise.allSettled(v.map(f=>f.close()))).filter(f=>f.status==="rejected").map(f=>new Error(f.reason));if($.length>0)throw new AggregateError($,"Failed to close some servers")}}export{N as DEFAULT_ACCELERATE_PORT,V as DEFAULT_DATABASE_PORT,K as DEFAULT_SHADOW_DATABASE_PORT,g as PortNotAvailableError,Gt as unstable_startServer};
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@prisma/dev",
3
- "version": "0.0.0-dev.202505192354",
3
+ "version": "0.0.0-dev.202505200036",
4
4
  "description": "A local Prisma Postgres server for development and testing",
5
5
  "type": "module",
6
6
  "main": "./dist/index.js",