@prisma/dev 0.0.0-dev.202505131254 → 0.0.0-dev.202505131401
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.js +4 -4
- package/package.json +2 -1
package/dist/index.js
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { createRequire } from 'module'; const require = createRequire(import.meta.url);
|
|
2
|
-
import{
|
|
3
|
-
`).find(S=>S.includes("Started query engine http server"));if(!c)return;n.stdout.removeListener("data",a);let{fields:
|
|
4
|
-
Received data: ${i}`));s({childProcess:n,url:`http://${
|
|
2
|
+
import{createSecureServer as Fe}from"http2";import{promisify as Ke}from"util";import{serve as Me}from"@hono/node-server";import Ve from"@prisma/get-platform";import{Hono as ce}from"hono/tiny";import{validator as y}from"hono/validator";import{HTTPException as P}from"hono/http-exception";import{object as le,optional as pe,parse as ue,pipe as B,regex as L,string as j,url as U}from"valibot";var q=/^(postgres|postgresql):\/\//,de=le({databaseUrl:B(j(),U(),L(q)),shadowDatabaseUrl:pe(B(j(),U(),L(q)))});function F(e){return Buffer.from(JSON.stringify(e),"utf8").toString("base64url")}function fe(e){try{let t=Buffer.from(e,"base64url").toString("utf8");return ue(de,JSON.parse(t))}catch{return null}}var p=(e,t)=>{let{authorization:n}=e;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=fe(o);if(a==null)throw new P(401,{message:"Invalid API Key"});let{databaseUrl:i}=a;if(i!==t.var.db.connectionString)throw new P(401,{message:"Unauthorized"});return{decodedAPIKey:a}};import{array as ge,literal as he,minLength as ye,object as we,pipe as Se,safeParse as Pe,string as be,union as ve}from"valibot";var Ee=we({tags:ve([Se(ge(be()),ye(1)),he("all")])});async function K(e){let{output:t,success:n}=Pe(Ee,await e.req.json());return n?t:e.text("Invalid input",400)}import{mkdir as Ce,readFile as N,writeFile as z}from"fs/promises";import{join as I}from"path";import{certificateFor as Oe}from"@expo/devcert";import{chmod as Te,stat as Ae,writeFile as Re}from"fs/promises";import xe from"env-paths";import{inflate as He}from"pako";var H=xe("prisma-dev").cache;function M(e,t){return`${H}/${e}/${t}`}async function g(e){try{return await Ae(e),!0}catch(t){if(t!=null&&typeof t=="object"&&"code"in t&&t.code==="ENOENT")return!1;throw t}}async function V(e,t){let n=He(e);await Re(t,n),await Te(t,"755")}var k=I(H,"certs"),C=I(k,"tls.cert"),O=I(k,"tls.key");async function G(){if(await Ce(k,{recursive:!0}),await g(C)&&await g(O))return{cert:await N(C),key:await N(O)};let e=await Oe("localhost");return await Promise.all([z(C,e.cert),z(O,e.key)]),{cert:e.cert,key:e.key}}import{spawn as Ie}from"child_process";import{once as ke}from"events";import{mkdir as _e}from"fs/promises";import{join as De}from"path";function b(){let e,t,n=new Promise((r,o)=>{e=r,t=o});return{fail:t,promise:n,succeed:e}}var v=class e{static#n=new Map;#t;#e;constructor(t){this.#t=t,this.#e=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),await o.start(),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.#e,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 u.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.#e,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 u.fromResponse(s);return await s.json()}async start(){if(this.#e!=null)return;let t=await this.#s(),n=Ie(t,["--enable-raw-queries","--enable-telemetry-in-response","--port","0"],{env:{LOG_QUERIES:"y",PRISMA_DML:this.#t.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}=b();this.#e=o;let a=i=>{let c=i.split(`
|
|
3
|
+
`).find(S=>S.includes("Started query engine http server"));if(!c)return;n.stdout.removeListener("data",a);let{fields:m}=JSON.parse(c);if(m==null)return r(new Error(`Unexpected data during initialization, "fields" are missing: ${i}`));let{ip:l,port:x}=m;if(l==null||x==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://${l}:${x}`})};n.once("error",i=>{this.#e=null,r(new h(String(i)))}),n.once("exit",(i,c)=>{this.#e=null,r(new h(`Query Engine exited with code ${i} and signal ${c}`))}),n.stdout.on("data",a),n.stderr.on("data",i=>{console.error(i)}),await this.#e}async stop(){if(this.#e==null)return;let{childProcess:t}=await this.#e;t.exitCode==null&&t.signalCode==null&&(t.kill(),await ke(t,"exit"))}async#s(){let t=await this.#a(),n=M(this.#t.clientVersion,t);await _e(n,{recursive:!0});let{platform:r}=this.#t.platform,o=r==="windows"?".exe":"",s=De(n,`query-engine-${r}${o}`);return await g(s)||await this.#i({commitHash:t,extension:o,engineBinaryPath:s}),s}async#a(){let t=await fetch(`https://registry.npmjs.org/@prisma/client/${this.#t.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,{platform:s}=this.#t.platform,a=`https://binaries.prisma.sh/all_commits/${n}/${s}/query-engine${r}.gz`,i=await fetch(a);if(!i.ok)throw new Error(`Couldn't download engine. URL: ${a}, status code: ${i.status}`);await V(await i.arrayBuffer(),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.#e,s=this.#r(n),a=await fetch(`${o}/transaction/${t}/${r}`,{headers:{...s,"Content-Type":"application/json"},method:"POST"});if(!a.ok)throw await u.fromResponse(a);try{return await a.json()}catch{return{}}}};function E(e,t){return console.error(e),e instanceof h?t.json({EngineNotStarted:{reason:{EngineStartupError:{logs:[],msg:e.message}}}},500):e instanceof u?t.text(e.responseBody,e.statusCode):t.body(null,500)}var h=class extends Error{name="EngineStartError"},u=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 J=51213,Q=51214,X=51215,d=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 W}from"buffer";var T=new Map;async function Y(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 $e=/datasource\s+db\s+[\w\s\d{="[\])(]+}/;function ee(e,t){let n=W.from(e,"base64").toString("utf8"),r=`datasource db {
|
|
5
5
|
provider = "postgresql"
|
|
6
6
|
url = "${t}"
|
|
7
|
-
}`,o=n.replace(
|
|
7
|
+
}`,o=n.replace($e,r);return W.from(o,"utf8").toString("base64")}function A(e){let{req:t}=e;return{traceparent:t.header("traceparent"),"X-capture-telemetry":t.header("X-capture-telemetry")}}import{integer as te,looseObject as Be,minValue as ne,number as _,object as Le,pipe as re,safeParse as oe,string as se,union as je}from"valibot";var Ue=Le({isolation_level:se(),max_wait:re(_(),te(),ne(0)),timeout:re(_(),te(),ne(0))});async function ae(e){let{output:t,success:n}=oe(Ue,await e.req.json());return n?t:e.json({EngineNotStarted:{reason:"InvalidRequest"}},400)}var qe=Be({id:je([se(),_()])});function ie(e,t){let{output:n,success:r}=oe(qe,e);return r?n:t.json({EngineMalfunction:{}},500)}async function me(e,t,n){let r=await Ne(e,t,n),o=await G(),{fail:s,promise:a,succeed:i}=b(),c=Me({createServer:Fe,fetch:r.fetch,port:e,serverOptions:{...o,allowHTTP1:!0}},()=>{i()});return c.on("error",m=>{if(typeof m=="object"&&"code"in m&&m.code==="EADDRINUSE")return s(new d(e,"accelerate"));console.error(m)}),await a,{async close(){await Ke(c.close.bind(c))()},port:e,url:`http://localhost:${e}`}}async function Ne(e,t,n){let r=new ce,o=await Ve.getPlatformInfo();return n&&console.debug("platform info: %s",JSON.stringify(o)),r.use("*",async(s,a)=>(s.set("db",t),s.set("debug",!!n),s.set("platform",o),s.set("port",e),s.set("protocol","https"),await a())),r.route("/",R),r}var R=new ce;R.post("/invalidate",y("header",p),async e=>{let t=await K(e);return t instanceof Response?t:e.body(null)});var ze="/:clientVersion/:schemaHash",w=R.basePath(ze);R.route("/",w);var Ge=["/graphql","/itx/:transactionId/graphql"];w.on("POST",[...Ge],y("header",p),async e=>{let{req:t}=e;try{let n=await D(e);if(console.log("got engine!"),n instanceof Response)return n;let r=await t.text(),o=t.param("transactionId"),s=await n.request(r,{...A(e),"X-transaction-id":o});return e.text(s)}catch(n){return E(n,e)}});w.basePath("/itx/:transactionId").on("POST",["/commit","/rollback"],y("header",p),async e=>{let{req:t}=e;try{let n=await D(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,A(e));return e.json(a)}catch(n){return E(n,e)}});w.put("/schema",y("header",p),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);return o==null?r!==await Y(n)?e.text("Schema hash mismatch",400):(T.set(r,{original:n,override:ee(n,e.get("db").connectionString)}),e.text(r)):n!==o.original?e.text("Schema mismatch",400):e.text(r)});w.post("/transaction/start",y("header",p),async e=>{let{req:t}=e,n=await ae(e);if(n instanceof Response)return n;try{let r=await D(e);if(r instanceof Response)return r;let o=await r.startTransaction(n,A(e)),s=ie(o,e);if(s instanceof Response)return s;let{id:a}=s,i=t.param("clientVersion"),c=e.get("port"),m=e.get("protocol"),l=t.param("schemaHash");return e.json({...o,"data-proxy":{endpoint:`${m}://localhost:${c}/${i}/${l}/itx/${a}`}})}catch(r){return E(r,e)}});async function D(e){let{req:t}=e,n=Z(e);return n instanceof Response?n:await v.get({base64Schema:n.schemas.override,clientVersion:t.param("clientVersion"),platform:e.get("platform"),schemaHash:n.schemaHash})}import{PGlite as Je}from"@electric-sql/pglite";import{PGLiteSocketServer as Qe}from"@electric-sql/pglite-socket";async function $(e,t,n){let r={connectionLimit:1,connectTimeout:0,database:"postgres",maxIdleConnectionLifetime:0,password:"postgres",poolTimeout:0,port:e,socketTimeout:0,sslMode:"disable",username:"postgres"},o=await Je.create({database:r.database,dataDir:"memory://",debug:n?5:void 0,defaultDataTransferContainer:"file",username:r.username});await o.waitReady;let s=new Qe({db:o,inspect:n,port:e});try{await s.start()}catch(i){throw i instanceof Error&&"code"in i&&i.code==="EADDRINUSE"?new d(e,t):i}n&&console.log(`${t} server started on port %i`,e);let a=`postgres://${r.username}:${r.password}@localhost:${e}/${r.database}?${new URLSearchParams({connection_limit:String(r.connectionLimit),connect_timeout:String(r.connectTimeout),max_idle_connection_lifetime:String(r.maxIdleConnectionLifetime),pool_timeout:String(r.poolTimeout),socket_timeout:String(r.socketTimeout),sslmode:r.sslMode}).toString()}`;return{...r,async close(){try{await s.stop()}finally{await o.close()}},connectionString:a}}async function Qt(e){let{acceleratePort:t=51213,databasePort:n=51214,debug:r=!1,shadowDatabasePort:o=51215}=e||{},[s,a]=await Promise.all([$(n,"database",r),$(o,"shadow_database",r)]),i=await me(t,s,r),c=`prisma+postgres://localhost:${i.port}/?${new URLSearchParams({api_key:F({databaseUrl:s.connectionString,shadowDatabaseUrl:a.connectionString})}).toString()}`;return{accelerate:{url:i.url},close:()=>m([i,s,a]),database:{connectionString:s.connectionString},ppg:{url:c},shadowDatabase:{connectionString:a.connectionString}};async function m(l){let S=(await Promise.allSettled(l.map(f=>f.close()))).filter(f=>f.status==="rejected").map(f=>new Error(f.reason));if(S.length>0)throw new AggregateError(S,"Failed to close some servers")}}export{J as DEFAULT_ACCELERATE_PORT,Q as DEFAULT_DATABASE_PORT,X as DEFAULT_SHADOW_DATABASE_PORT,d as PortNotAvailableError,Qt as unstable_startServer};
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@prisma/dev",
|
|
3
|
-
"version": "0.0.0-dev.
|
|
3
|
+
"version": "0.0.0-dev.202505131401",
|
|
4
4
|
"description": "A local Prisma Postgres server for development and testing",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "./dist/index.js",
|
|
@@ -38,6 +38,7 @@
|
|
|
38
38
|
"dependencies": {
|
|
39
39
|
"@electric-sql/pglite": "0.3.1",
|
|
40
40
|
"@electric-sql/pglite-socket": "0.0.4",
|
|
41
|
+
"@expo/devcert": "1.2.0",
|
|
41
42
|
"@hono/node-server": "1.14.1",
|
|
42
43
|
"@prisma/get-platform": "6.7.0",
|
|
43
44
|
"env-paths": "3.0.0",
|