@prisma/dev 0.0.0-dev.202505131416 → 0.0.0-dev.202505131838
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 +3 -3
- package/package.json +1 -1
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 $e}from"http";import{promisify as
|
|
2
|
+
import{createServer as $e}from"http";import{promisify as ke}from"util";import{serve as De}from"@hono/node-server";import _e from"@prisma/get-platform";import{logger as Be}from"hono/logger";import{Hono as ee}from"hono/tiny";import{validator as h}from"hono/validator";import{HTTPException as b}from"hono/http-exception";import{object as ne,optional as re,parse as oe,pipe as C,regex as $,string as k,url as D}from"valibot";var _=/^(postgres|postgresql):\/\//,se=ne({databaseUrl:C(k(),D(),$(_)),shadowDatabaseUrl:re(C(k(),D(),$(_)))});function B(e){return Buffer.from(JSON.stringify(e),"utf8").toString("base64url")}function ae(e){try{let t=Buffer.from(e,"base64url").toString("utf8");return oe(se,JSON.parse(t))}catch{return null}}var p=(e,t)=>{let{authorization:n}=e;if(!n)throw new b(401,{message:"Missing API Key"});let[r,o="",s]=n.split(" ");if(r!=="Bearer"||s)throw new b(401,{message:"Invalid API Key"});let a=ae(o);if(a==null)throw new b(401,{message:"Invalid API Key"});let{databaseUrl:i}=a;if(i!==t.var.db.connectionString)throw new b(401,{message:"Unauthorized"});return{decodedAPIKey:a}};import{array as ie,literal as ce,minLength as le,object as me,pipe as pe,safeParse as de,string as ue,union as fe}from"valibot";var ge=me({tags:fe([pe(ie(ue()),le(1)),ce("all")])});async function L(e){let{output:t,success:n}=de(ge,await e.req.json());return n?t:e.text("Invalid input",400)}import{spawn as ve}from"child_process";import{once as Ee}from"events";import{mkdir as Ae}from"fs/promises";import{join as Te}from"path";import{chmod as he,stat as ye,writeFile as we}from"fs/promises";import Se from"env-paths";import{inflate as be}from"pako";var Pe=Se("prisma-dev").cache;function j(e,t){return`${Pe}/${e}/${t}`}async function U(e){try{return await ye(e),!0}catch(t){if(t!=null&&typeof t=="object"&&"code"in t&&t.code==="ENOENT")return!1;throw t}}async function q(e,t){let n=be(e);await we(t,n),await he(t,"755")}function P(){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),t.debug&&console.debug("starting engine",t),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 d.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 d.fromResponse(s);return await s.json()}async start(){if(this.#e!=null)return;let t=await this.#s(),n=ve(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}=P();this.#e=o;let a=i=>{let c=i.split(`
|
|
3
3
|
`).find(S=>S.includes("Started query engine http server"));if(!c)return;n.stdout.removeListener("data",a);let{fields:l}=JSON.parse(c);if(l==null)return r(new Error(`Unexpected data during initialization, "fields" are missing: ${i}`));let{ip:m,port:x}=l;if(m==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://${m}:${x}`})};n.once("error",i=>{this.#e=null,r(new g(String(i)))}),n.once("exit",(i,c)=>{this.#e=null,r(new g(`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 Ee(t,"exit"))}async#s(){let t=await this.#a(),n=
|
|
4
|
+
Received data: ${i}`));s({childProcess:n,url:`http://${m}:${x}`})};n.once("error",i=>{this.#e=null,r(new g(String(i)))}),n.once("exit",(i,c)=>{this.#e=null,r(new g(`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 Ee(t,"exit"))}async#s(){let t=await this.#a(),n=j(this.#t.clientVersion,t);await Ae(n,{recursive:!0});let{platform:r}=this.#t.platform,o=r==="windows"?".exe":"",s=Te(n,`query-engine-${r}${o}`);return await U(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 q(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 d.fromResponse(a);try{return await a.json()}catch{return{}}}};function E(e,t){return console.error(e),e instanceof g?t.json({EngineNotStarted:{reason:{EngineStartupError:{logs:[],msg:e.message}}}},500):e instanceof d?t.text(e.responseBody,e.statusCode):t.body(null,500)}var g=class extends Error{name="EngineStartError"},d=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 F=51213,K=51214,M=51215,u=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 V}from"buffer";var A=new Map;async function R(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 N(e){let t=e.req.param("schemaHash"),n=A.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 z(e,t){let n=V.from(e,"base64").toString("utf8");console.log("Schema: ",n);let r=`datasource db {
|
|
5
5
|
provider = "postgresql"
|
|
6
6
|
url = "${t}"
|
|
7
|
-
}`,o=n.replace(xe,r);return
|
|
7
|
+
}`,o=n.replace(xe,r);console.log("Schema with override: ",o);let s=await R(o);return{base64Override:V.from(o,"utf8").toString("base64"),overrideHash:s}}function T(e){let{req:t}=e;return{traceparent:t.header("traceparent"),"X-capture-telemetry":t.header("X-capture-telemetry")}}import{integer as G,looseObject as Re,minValue as J,number as H,object as He,pipe as Q,safeParse as X,string as W,union as Oe}from"valibot";var Ie=He({isolation_level:W(),max_wait:Q(H(),G(),J(0)),timeout:Q(H(),G(),J(0))});async function Y(e){let{output:t,success:n}=X(Ie,await e.req.json());return n?t:e.json({EngineNotStarted:{reason:"InvalidRequest"}},400)}var Ce=Re({id:Oe([W(),H()])});function Z(e,t){let{output:n,success:r}=X(Ce,e);return r?n:t.json({EngineMalfunction:{}},500)}async function te(e,t,n){let r=await Le(e,t,n),{fail:o,promise:s,succeed:a}=P(),i=De({createServer:$e,fetch:r.fetch,port:e},()=>{a()});return i.on("error",c=>{if(typeof c=="object"&&"code"in c&&c.code==="EADDRINUSE")return o(new u(e,"accelerate"));console.error(c)}),await s,{async close(){await ke(i.close.bind(i))()},port:e,url:`http://localhost:${e}`}}async function Le(e,t,n){let r=new ee,o=await _e.getPlatformInfo();return n&&console.debug("platform info: %s",JSON.stringify(o)),n&&r.use("*",Be()),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("/",y),r}var y=new ee;y.get("/health",e=>e.text("ok"));y.post("/invalidate",h("header",p),async e=>{let t=await L(e);return t instanceof Response?t:e.body(null)});var je="/:clientVersion/:schemaHash",w=y.basePath(je);y.route("/",w);var Ue=["/graphql","/itx/:transactionId/graphql"];w.on("POST",[...Ue],h("header",p),async e=>{let{req:t}=e;try{let n=await O(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,{...T(e),"X-transaction-id":o});return e.text(s)}catch(n){return E(n,e)}});w.basePath("/itx/:transactionId").on("POST",["/commit","/rollback"],h("header",p),async e=>{let{req:t}=e;try{let n=await O(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,T(e));return e.json(a)}catch(n){return E(n,e)}});w.put("/schema",h("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=A.get(r);if(o==null){if(r!==await R(n))return e.text("Schema hash mismatch",400);let s=await z(n,e.get("db").connectionString);return A.set(r,{base64Original:n,...s}),e.text(r)}return n!==o.base64Original?e.text("Schema mismatch",400):e.text(r)});w.post("/transaction/start",h("header",p),async e=>{let{req:t}=e,n=await Y(e);if(n instanceof Response)return n;try{let r=await O(e);if(r instanceof Response)return r;let o=await r.startTransaction(n,T(e)),s=Z(o,e);if(s instanceof Response)return s;let{id:a}=s,i=t.param("clientVersion"),c=e.get("port"),l=e.get("protocol"),m=t.param("schemaHash");return e.json({...o,"data-proxy":{endpoint:`${l}://localhost:${c}/${i}/${m}/itx/${a}`}})}catch(r){return E(r,e)}});async function O(e){let{req:t}=e,n=N(e);if(n instanceof Response)return n;let{base64Override:r,overrideHash:o}=n.schemas;return await v.get({base64Schema:r,clientVersion:t.param("clientVersion"),debug:e.get("debug"),platform:e.get("platform"),schemaHash:o})}import{PGlite as qe}from"@electric-sql/pglite";import{PGLiteSocketServer as Fe}from"@electric-sql/pglite-socket";async function I(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 qe.create({database:r.database,dataDir:"memory://",debug:n?5:void 0,defaultDataTransferContainer:"file",username:r.username});await o.waitReady;let s=new Fe({db:o,inspect:n,port:e});try{await s.start()}catch(i){throw i instanceof Error&&"code"in i&&i.code==="EADDRINUSE"?new u(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 Bt(e){let{acceleratePort:t=51213,databasePort:n=51214,debug:r=!1,shadowDatabasePort:o=51215}=e||{},[s,a]=await Promise.all([I(n,"database",r),I(o,"shadow_database",r)]),i=await te(t,s,r),c=`prisma+postgres://localhost:${i.port}/?${new URLSearchParams({api_key:B({databaseUrl:s.connectionString,shadowDatabaseUrl:a.connectionString})}).toString()}`;return{accelerate:{url:i.url},close:()=>l([i,s,a]),database:{connectionString:s.connectionString},ppg:{url:c},shadowDatabase:{connectionString:a.connectionString}};async function l(m){let S=(await Promise.allSettled(m.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{F as DEFAULT_ACCELERATE_PORT,K as DEFAULT_DATABASE_PORT,M as DEFAULT_SHADOW_DATABASE_PORT,u as PortNotAvailableError,Bt as unstable_startServer};
|