@prisma/dev 0.0.0-dev.202505220029 → 0.0.0-dev.202505220316

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
@@ -1,25 +1,4 @@
1
- type DBPersistenceMode = "dump_on_close" | "memory";
2
-
3
- declare const DEFAULT_ACCELERATE_PORT = 51213;
4
- declare const DEFAULT_DATABASE_PORT = 51214;
5
- declare const DEFAULT_SHADOW_DATABASE_PORT = 51215;
6
- type PortAssignableService = "accelerate" | "database" | "shadow_database";
7
- declare class PortNotAvailableError extends Error {
8
- port: number;
9
- service: PortAssignableService;
10
- name: string;
11
- constructor(port: number, service: PortAssignableService);
12
- }
13
-
14
- interface StartServerOptions {
15
- /**
16
- * The port the Accelerate server will listen on.
17
- *
18
- * Defaults to `51213`.
19
- *
20
- * An error is thrown if the port is already in use.
21
- */
22
- acceleratePort?: number;
1
+ interface ServerOptions {
23
2
  /**
24
3
  * The port the database server will listen on.
25
4
  *
@@ -41,11 +20,25 @@ interface StartServerOptions {
41
20
  */
42
21
  dryRun?: boolean;
43
22
  /**
44
- * TODO: Add description.
23
+ * The name of the server.
24
+ *
25
+ * Defaults to `default`.
26
+ */
27
+ name?: string;
28
+ /**
29
+ * The persistence mode of the server.
45
30
  *
46
- * Default is `memory`.
31
+ * Default is `stateless`.
47
32
  */
48
- persistenceMode?: DBPersistenceMode;
33
+ persistenceMode?: PersistenceMode;
34
+ /**
35
+ * The port the server will listen on.
36
+ *
37
+ * Defaults to `51213`.
38
+ *
39
+ * An error is thrown if the port is already in use.
40
+ */
41
+ port?: number;
49
42
  /**
50
43
  * The port the shadow database server will listen on.
51
44
  *
@@ -55,6 +48,21 @@ interface StartServerOptions {
55
48
  */
56
49
  shadowDatabasePort?: number;
57
50
  }
51
+ type PersistenceMode = "stateless" | "stateful";
52
+
53
+ type DBServerPurpose = "database" | "shadow_database";
54
+
55
+ declare const DEFAULT_DATABASE_PORT = 51214;
56
+ declare const DEFAULT_SERVER_PORT = 51213;
57
+ declare const DEFAULT_SHADOW_DATABASE_PORT = 51215;
58
+ type PortAssignableService = DBServerPurpose | "server";
59
+ declare class PortNotAvailableError extends Error {
60
+ port: number;
61
+ service: PortAssignableService;
62
+ name: string;
63
+ constructor(port: number, service: PortAssignableService);
64
+ }
65
+
58
66
  interface Server {
59
67
  accelerate: {
60
68
  url: string;
@@ -70,6 +78,6 @@ interface Server {
70
78
  connectionString: string;
71
79
  };
72
80
  }
73
- declare function unstable_startServer(options?: StartServerOptions): Promise<Server>;
81
+ declare function unstable_startServer(options?: ServerOptions): Promise<Server>;
74
82
 
75
- export { DEFAULT_ACCELERATE_PORT, DEFAULT_DATABASE_PORT, DEFAULT_SHADOW_DATABASE_PORT, type PortAssignableService, PortNotAvailableError, type Server, type StartServerOptions, unstable_startServer };
83
+ export { DEFAULT_DATABASE_PORT, DEFAULT_SERVER_PORT, DEFAULT_SHADOW_DATABASE_PORT, type PortAssignableService, PortNotAvailableError, type Server, unstable_startServer };
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 qe}from"http";import{promisify as Ne}from"util";import{serve as Fe}from"@hono/node-server";import Ve from"@prisma/get-platform";import{logger as Ke}from"hono/logger";import{Hono as oe}from"hono/tiny";import{validator as w}from"hono/validator";import{HTTPException as P}from"hono/http-exception";import{object as ie,optional as ce,parseJson as pe,pipe as x,regex as C,safeParse as ue,string as D,url as k}from"valibot";var L=/^(postgres|postgresql):\/\//,le=x(D(),pe(),ie({databaseUrl:x(D(),k(),C(L)),shadowDatabaseUrl:ce(x(D(),k(),C(L)))}));function M(t){return Buffer.from(JSON.stringify(t),"utf8").toString("base64url")}function de(t){let e=Buffer.from(t,"base64url").toString("utf8"),{issues:r,output:n,success:o}=ue(le,e,{abortEarly:!0});return o?[null,n]:[r]}var d=(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]=de(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 me,literal as ge,minLength as fe,object as he,pipe as ye,safeParse as we,string as Se,union as be}from"valibot";var ve=he({tags:be([ye(me(Se()),fe(1)),ge("all")])});async function U(t){let{output:e,success:r}=we(ve,await t.req.json(),{abortEarly:!0});return r?e:t.text("Invalid input",400)}import{spawn as _e}from"child_process";import{once as He}from"events";import{mkdir as Ie}from"fs/promises";import{join as $e}from"path";import{createWriteStream as Pe,WriteStream as Ee}from"fs";import{chmod as Ae,stat as Te,writeFile as Re}from"fs/promises";import Oe from"env-paths";import{inflate as xe}from"pako";var De=Oe("prisma-dev");function j(t,e){return`${De.cache}/engine/${t}/${e}`}async function q(t){try{return await Te(t),!0}catch(e){if(e!=null&&typeof e=="object"&&"code"in e&&e.code==="ENOENT")return!1;throw e}}async function N(t,e){let r=xe(t);await Re(e,r),await Ae(e,"755")}async function F(t,e){await t.stream().pipeTo(Ee.toWeb(Pe(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 h=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 m.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 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 r=_e(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(O=>O.includes("Started query engine http server"));if(!c)return;r.stdout.removeListener("data",a);let{fields:u}=JSON.parse(c);if(u==null)return n(new Error(`Unexpected data during initialization, "fields" are missing: ${i}`));let{ip:l,port:v}=u;if(l==null||v==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://${l}:${v}`})};r.once("error",i=>{this.#t=null,n(new y(String(i)))}),r.once("exit",(i,c)=>{this.#t=null,n(new y(`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 He(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=j(this.#e.clientVersion,e);this.#e.debug&&console.debug("cache directory path",r),await Ie(r,{recursive:!0});let{platform:n}=this.#e.platform,o=n==="windows"?".exe":"",s=$e(r,`query-engine-${n}${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: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 N(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 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(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 V=51213,K=51214,G=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 z}from"buffer";var T=new Map;async function _(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 J(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 Be=/datasource\s+db\s+\{\s*provider\s*=\s*"postgres(!?ql)?"\s+url\s*=\s*.+\s*\}/;async function W(t,e){let r=z.from(t,"base64").toString("utf8"),n=`datasource db {
2
+ import{createServer as Ve}from"http";import{promisify as Ke}from"util";import{serve as Ge}from"@hono/node-server";import ze from"@prisma/get-platform";import{logger as Je}from"hono/logger";import{Hono as ie}from"hono/tiny";import{validator as b}from"hono/validator";import{HTTPException as P}from"hono/http-exception";import{object as le,optional as pe,parseJson as me,pipe as R,regex as C,safeParse as ge,string as D,url as L}from"valibot";var M=/^(postgres|postgresql):\/\//,fe=R(D(),me(),le({databaseUrl:R(D(),L(),C(M)),shadowDatabaseUrl:pe(R(D(),L(),C(M)))}));function j(t){return Buffer.from(JSON.stringify(t),"utf8").toString("base64url")}function he(t){let e=Buffer.from(t,"base64url").toString("utf8"),{issues:r,output:n,success:o}=ge(fe,e,{abortEarly:!0});return o?[null,n]:[r]}var p=(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]=he(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 ye,literal as be,minLength as we,object as Se,pipe as Pe,safeParse as ve,string as Ee,union as Te}from"valibot";var Ae=Se({tags:Te([Pe(ye(Ee()),we(1)),be("all")])});async function U(t){let{output:e,success:r}=ve(Ae,await t.req.json(),{abortEarly:!0});return r?e:t.text("Invalid input",400)}import{spawn as He}from"child_process";import{once as ke}from"events";import{mkdir as Be}from"fs/promises";import{join as Ce}from"path";import{createWriteStream as Re,WriteStream as De}from"fs";import{chmod as Oe,stat as xe,writeFile as Ie}from"fs/promises";import _e from"env-paths";import{inflate as $e}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 xe(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=$e(t);await Ie(e,r),await Oe(e,"755")}async function G(t,e){await t.stream().pipeTo(De.toWeb(Re(e,{encoding:"utf-8"})))}function v(){let t,e,r=new Promise((n,o)=>{t=n,e=o});return{fail:e,promise:r,succeed:t}}var h=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 m.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 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 r=He(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}=v();this.#t=o;let a=i=>{let c=i.split(`
3
+ `).find(ue=>ue.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:u,port:B}=l;if(u==null||B==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://${u}:${B}`})};r.once("error",i=>{this.#t=null,n(new y(String(i)))}),r.once("exit",(i,c)=>{this.#t=null,n(new y(`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 ke(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 Be(r,{recursive:!0});let{platform:n}=this.#e.platform,o=n==="windows"?".exe":"",s=Ce(r,`query-engine-${n}${o}`);return this.#e.debug&&console.debug("engine binary path",s),(process.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 m.fromResponse(a);try{return await a.json()}catch{return{}}}};function E(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(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 O(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 Le=/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 {
5
5
  provider = "postgresql"
6
6
  url = "${e}"
7
- }`,o=r.replace(Be,n),s=await _(o);return{base64Override:z.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 Ce,minValue as X,number as H,object as ke,optional as Le,pipe as Y,safeParse as Z,string as ee,union as Me}from"valibot";var Ue=ke({isolation_level:Le(ee()),max_wait:Y(H(),Q(),X(0)),timeout:Y(H(),Q(),X(0))});async function te(t){let{issues:e,output:r,success:n}=Z(Ue,await t.req.json(),{abortEarly:!0});return n?r:t.json({EngineNotStarted:{reason:"InvalidRequest",issues:e}},400)}var je=Ce({id:Me([ee(),H()])});function re(t,e){let{output:r,success:n}=Z(je,t);return n?r:e.json({EngineMalfunction:{}},500)}async function se(t){let{dbServer:e,debug:r,dryRun:n,port:o}=t;if(n)return ne(o,null);let s=await Ge(o,e,r),{fail:a,promise:i,succeed:c}=E(),u=Fe({createServer:qe,fetch:s.fetch,port:o},()=>{c()});return u.on("error",l=>{if(typeof l=="object"&&"code"in l&&l.code==="EADDRINUSE")return a(new g(o,"accelerate"));console.error(l)}),await i,ne(o,u)}function ne(t,e){return{async close(){e&&await Promise.allSettled([Ne(e.close.bind(e))(),h.stopAll()])},port:t,url:`http://localhost:${t}`}}async function Ge(t,e,r){let n=new oe,o=await Ve.getPlatformInfo();return r&&console.debug("platform info: %s",JSON.stringify(o)),r&&n.use("*",Ke()),n.use("*",async(s,a)=>(s.set("db",e),s.set("debug",!!r),s.set("platform",o),s.set("port",t),s.set("protocol","http"),await a())),n.route("/",S),n}var S=new oe;S.get("/health",t=>t.text("ok"));S.post("/invalidate",w("header",d),async t=>{let e=await U(t);return e instanceof Response?e:t.body(null)});var ze="/:clientVersion/:schemaHash",b=S.basePath(ze);S.route("/",b);var Je=["/graphql","/itx/:transactionId/graphql"];b.on("POST",[...Je],w("header",d),async t=>{let{req:e}=t;try{let r=await I(t);if(r instanceof Response)return r;let n=await e.text(),o=e.param("transactionId"),s=await r.request(n,{...R(t),"X-transaction-id":o});return t.text(s)}catch(r){return A(r,t)}});b.basePath("/itx/:transactionId").on("POST",["/commit","/rollback"],w("header",d),async t=>{let{req:e}=t;try{let r=await I(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,R(t));return t.json(a)}catch(r){return A(r,t)}});b.put("/schema",w("header",d),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 _(r))return t.text("Schema hash mismatch",400);let s=await W(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)});b.post("/transaction/start",w("header",d),async t=>{let{req:e}=t,r=await te(t);if(r instanceof Response)return r;try{let n=await I(t);if(n instanceof Response)return n;let o=await n.startTransaction(r,R(t)),s=re(o,t);if(s instanceof Response)return s;let{id:a}=s,i=e.param("clientVersion"),c=t.get("port"),u=t.get("protocol"),l=e.param("schemaHash");return t.json({...o,"data-proxy":{endpoint:`${u}://localhost:${c}/${i}/${l}/itx/${a}`}})}catch(n){return A(n,t)}});async function I(t){let{req:e}=t,r=J(t);if(r instanceof Response)return r;let{base64Override:n,overrideHash:o}=r.schemas;return await h.get({base64Schema:n,clientVersion:process.env.PRISMA_DEV_FORCE_CLIENT_VERSION||e.param("clientVersion"),debug:t.get("debug"),platform:t.get("platform"),schemaHash:o})}import{copyFile as We,readFile as Qe,unlink as Xe}from"fs/promises";import{PGlite as Ye}from"@electric-sql/pglite";import{PGLiteSocketServer as Ze}from"@electric-sql/pglite-socket";import{pgDump as et}from"@electric-sql/pglite-tools/pg_dump";var p={connectionLimit:1,connectTimeout:0,database:"postgres",maxIdleConnectionLifetime:0,password:"postgres",poolTimeout:0,socketTimeout:0,sslMode:"disable",username:"postgres"},tt=`postgres://${p.username}:${p.password}@localhost`,rt=new URLSearchParams({connection_limit:String(p.connectionLimit),connect_timeout:String(p.connectTimeout),max_idle_connection_lifetime:String(p.maxIdleConnectionLifetime),pool_timeout:String(p.poolTimeout),socket_timeout:String(p.socketTimeout),sslmode:p.sslMode});async function $(t){let{debug:e,dryRun:r,persistenceMode:n,port:o,purpose:s}=t;if(r)return ae({...t,db:null,server:null});let a=await Ye.create({database:p.database,dataDir:"memory://",debug:e?5:void 0,defaultDataTransferContainer:"file",username:p.username});await a.waitReady,n==="dump_on_close"&&await nt(a);let i=new Ze({db:a,inspect:e,port:o});try{await i.start()}catch(c){throw c instanceof Error&&"code"in c&&c.code==="EADDRINUSE"?new g(o,s):c}return e&&console.debug(`${s} server started on port %i`,o),ae({...t,db:a,server:i})}function ae(t){let{db:e,debug:r,persistenceMode:n,port:o,purpose:s,server:a}=t;return{...p,close:async()=>{if(!e||!a)return;let i=[];try{await a.stop(),r&&console.debug(`${s} server stopped on port %i`,o)}catch(c){console.error("server stop error",c),i.push(c)}if(n==="dump_on_close")try{await ot(e),r&&console.debug(`${s} state saved`)}catch(c){console.error("dump error",c),i.push(c)}try{await e.close(),r&&console.debug(`${s} closed`)}catch(c){console.error("db close error",c),i.push(c)}if(i.length>0)throw new AggregateError(i,`Failed to close ${s} properly`)},connectionString:st(o),port:o}}async function nt(t){try{let e=await Qe("./dump.sql",{encoding:"utf-8"});if(!e.length)return;await t.exec(`${e}; set search_path to public;`)}catch(e){if(e instanceof Error&&"code"in e&&e.code==="ENOENT")return console.warn("No database dump found. Starting with an empty database.");throw e}}async function ot(t){let e=!1;try{await We("./dump.sql","./dump.sql.bak"),e=!0}catch(n){if(!(n instanceof Error&&"code"in n&&n.code==="ENOENT"))throw n}let r=await et({args:[],pg:t});if(await F(r,"./dump.sql"),!!e)try{await Xe("./dump.sql.bak")}catch{}}function st(t){return`${tt}:${t}/${p.database}?${rt.toString()}`}async function ar(t){let{acceleratePort:e=51213,databasePort:r=51214,debug:n=!1,dryRun:o=!1,persistenceMode:s="memory",shadowDatabasePort:a=51215}=t||{},[i,c]=await Promise.all([$({debug:n,dryRun:o,persistenceMode:s,port:r,purpose:"database"}),$({debug:n,dryRun:o,persistenceMode:"memory",port:a,purpose:"shadow_database"})]),u=await se({dbServer:i,debug:n,dryRun:o,port:e}),l=`prisma+postgres://localhost:${u.port}/?${new URLSearchParams({api_key:M({databaseUrl:i.connectionString,shadowDatabaseUrl:c.connectionString})}).toString()}`;return{accelerate:{url:u.url},close:()=>v([u,i,c]),database:{connectionString:i.connectionString},ppg:{url:l},shadowDatabase:{connectionString:c.connectionString}};async function v(O){let B=(await Promise.allSettled(O.map(f=>f.close()))).filter(f=>f.status==="rejected").map(f=>new Error(f.reason));if(B.length>0)throw new AggregateError(B,"Failed to close some servers")}}export{V as DEFAULT_ACCELERATE_PORT,K as DEFAULT_DATABASE_PORT,G as DEFAULT_SHADOW_DATABASE_PORT,g as PortNotAvailableError,ar as unstable_startServer};
7
+ }`,o=r.replace(Le,n),s=await O(o);return{base64Override:Q.from(o,"utf8").toString("base64"),overrideHash:s}}function A(t){let{req:e}=t;return{traceparent:e.header("traceparent"),"X-capture-telemetry":e.header("X-capture-telemetry")}}import{integer as Z,looseObject as Me,minValue as ee,number as x,object as je,optional as Ue,pipe as te,safeParse as re,string as ne,union as Fe}from"valibot";var Ne=je({isolation_level:Ue(ne()),max_wait:te(x(),Z(),ee(0)),timeout:te(x(),Z(),ee(0))});async function oe(t){let{issues:e,output:r,success:n}=re(Ne,await t.req.json(),{abortEarly:!0});return n?r:t.json({EngineNotStarted:{reason:"InvalidRequest",issues:e}},400)}var qe=Me({id:Fe([ne(),x()])});function se(t,e){let{output:r,success:n}=re(qe,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 We(r,t,e),{fail:o,promise:s,succeed:a}=v(),i=Ge({createServer:Ve,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([Ke(e.close.bind(e))(),h.stopAll()])},port:t,url:`http://localhost:${t}`}}async function We(t,e,r){let{debug:n}=r,o=new ie,s=await ze.getPlatformInfo();return n&&console.debug("platform info: %s",JSON.stringify(s)),n&&o.use("*",Je()),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("/",w),o}var w=new ie;w.get("/health",t=>t.json({name:t.get("serverState").name}));w.post("/invalidate",b("header",p),async t=>{let e=await U(t);return e instanceof Response?e:t.body(null)});var Qe="/:clientVersion/:schemaHash",S=w.basePath(Qe);w.route("/",S);var Xe=["/graphql","/itx/:transactionId/graphql"];S.on("POST",[...Xe],b("header",p),async t=>{let{req:e}=t;try{let r=await I(t);if(r instanceof Response)return r;let n=await e.text(),o=e.param("transactionId"),s=await r.request(n,{...A(t),"X-transaction-id":o});return t.text(s)}catch(r){return E(r,t)}});S.basePath("/itx/:transactionId").on("POST",["/commit","/rollback"],b("header",p),async t=>{let{req:e}=t;try{let r=await I(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,A(t));return t.json(a)}catch(r){return E(r,t)}});S.put("/schema",b("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=T.get(n);if(o==null){if(n!==await O(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)});S.post("/transaction/start",b("header",p),async t=>{let{req:e}=t,r=await oe(t);if(r instanceof Response)return r;try{let n=await I(t);if(n instanceof Response)return n;let o=await n.startTransaction(r,A(t)),s=se(o,t);if(s instanceof Response)return s;let{id:a}=s,i=e.param("clientVersion"),c=t.get("port"),l=t.get("protocol"),u=e.param("schemaHash");return t.json({...o,"data-proxy":{endpoint:`${l}://localhost:${c}/${i}/${u}/itx/${a}`}})}catch(n){return E(n,t)}});async function I(t){let{req:e}=t,r=X(t);if(r instanceof Response)return r;let{base64Override:n,overrideHash:o}=r.schemas;return await h.get({base64Schema:n,clientVersion:process.env.PRISMA_DEV_FORCE_CLIENT_VERSION||e.param("clientVersion"),debug:t.get("debug"),platform:t.get("platform"),schemaHash:o})}import{PGlite as Ye}from"@electric-sql/pglite";import{PGLiteSocketServer as Ze}from"@electric-sql/pglite-socket";import{pgDump as et}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"},tt=`postgres://${d.username}:${d.password}@localhost`,rt=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 Ye.create({database:d.database,dataDir:"memory://",debug:n?5:void 0,defaultDataTransferContainer:"file",username:d.username});await o.waitReady,t!=="shadow_database"&&await nt(o,e);let s=new Ze({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 ot(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:st(s),port:s}}async function nt(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 ot(t,e){await e.writeDatabaseDump(()=>et({args:[],pg:t}))}function st(t){return`${tt}:${t}/${d.database}?${rt.toString()}`}import{copyFile as at,mkdir as it,readFile as ct,unlink as dt}from"fs/promises";import{join as ut}from"pathe";var k=Symbol("initialize"),f=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 H(e):new $(e);return await r[k](),r}},$=class extends f{constructor(e){super({...e,persistenceMode:"stateless"})}async[k](){}readDatabaseDump(){return Promise.resolve("")}async writeDatabaseDump(){}},H=class extends f{#r;#e;constructor(e){super({...e,persistenceMode:"stateful"}),this.#r=q(this.name),this.#e=ut(this.#r,"dump.sql")}async[k](){await it(this.#r,{recursive:!0}),this.debug&&console.log(`using data directory: ${this.#r}`)}async readDatabaseDump(){return await ct(this.#e,{encoding:"utf-8"})}async writeDatabaseDump(e){let r=!1,n=`${this.#e}.bak`;try{await at(this.#e,n),r=!0,this.debug&&console.log(`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.log(`dumped database to: ${this.#e}`),!!r)try{await dt(`${this.#e}.bak`),this.debug&&console.log(`removed backup dump: ${n}`)}catch{}}};async function mr(t){let e=await f.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()}`;return{accelerate:{url:o.url},close:()=>a([o,r,n]),database:{connectionString:r.connectionString},ppg:{url:s},shadowDatabase:{connectionString:n.connectionString}};async function a(i){let l=(await Promise.allSettled(i.map(u=>u.close()))).filter(u=>u.status==="rejected").map(u=>new Error(u.reason));if(l.length>0)throw new AggregateError(l,"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,mr as unstable_startServer};
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@prisma/dev",
3
- "version": "0.0.0-dev.202505220029",
3
+ "version": "0.0.0-dev.202505220316",
4
4
  "description": "A local Prisma Postgres server for development and testing",
5
5
  "type": "module",
6
6
  "main": "./dist/index.js",
@@ -44,6 +44,7 @@
44
44
  "env-paths": "3.0.0",
45
45
  "hono": "4.7.8",
46
46
  "pako": "2.1.0",
47
+ "pathe": "2.0.3",
47
48
  "valibot": "1.1.0"
48
49
  },
49
50
  "scripts": {