@prisma/dev 0.0.0-dev.202505221232 → 0.0.0-dev.202505221234
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 +5 -5
- package/package.json +2 -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
|
|
3
|
-
`).find(
|
|
4
|
-
Received data: ${i}`));s({childProcess:
|
|
2
|
+
import{createServer as We}from"http";import{promisify as Xe}from"util";import{serve as Ye}from"@hono/node-server";import Ze from"@prisma/get-platform";import{logger as et}from"hono/logger";import{Hono as de}from"hono/tiny";import{validator as S}from"hono/validator";import{process as tt}from"std-env";import{HTTPException as E}from"hono/http-exception";import{object as ge,optional as he,parseJson as fe,pipe as $,regex as j,safeParse as be,string as I,url as F}from"valibot";var U=/^(postgres|postgresql):\/\//,ye=$(I(),fe(),ge({databaseUrl:$(I(),F(),j(U)),shadowDatabaseUrl:he($(I(),F(),j(U)))}));function N(r){return Buffer.from(JSON.stringify(r),"utf8").toString("base64url")}function we(r){let e=Buffer.from(r,"base64url").toString("utf8"),{issues:t,output:n,success:o}=be(ye,e,{abortEarly:!0});return o?[null,n]:[t]}var p=(r,e)=>{let{authorization:t}=r;if(!t)throw new E(401,{message:"Missing API Key"});let[n,o="",s]=t.split(" ");if(n!=="Bearer"||s)throw new E(401,{message:"Invalid API Key"});let[a,i]=we(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 Se,literal as Pe,minLength as ve,object as Ee,pipe as De,safeParse as Ae,string as Te,union as Re}from"valibot";var Oe=Ee({tags:Re([De(Se(Te()),ve(1)),Pe("all")])});async function q(r){let{output:e,success:t}=Ae(Oe,await r.req.json(),{abortEarly:!0});return t?e:r.text("Invalid input",400)}import{spawn as Me}from"child_process";import{once as je}from"events";import{mkdir as Fe}from"fs/promises";import{join as Ue}from"path";import{process as Ne}from"std-env";import{createWriteStream as xe,WriteStream as $e}from"fs";import{access as Ie,chmod as ke,constants as _e,unlink as He,writeFile as Be}from"fs/promises";import Ce from"env-paths";import{inflate as Le}from"pako";var V=Ce("prisma-dev");function K(r,e){return`${V.cache}/engine/${r}/${e}`}function G(r){return`${V.data}/${r}`}async function D(r){try{return await Ie(r,_e.F_OK),!0}catch(e){if(e!=null&&typeof e=="object"&&"code"in e&&e.code==="ENOENT")return!1;throw e}}async function z(r,e){let t=Le(r);await Be(e,t),await ke(e,"755")}async function J(r,e){await r.stream().pipeTo($e.toWeb(xe(e,{encoding:"utf-8"})))}async function A(r){try{return await He(r),!0}catch{return!1}}function T(){let r,e,t=new Promise((n,o)=>{r=n,e=o});return{fail:e,promise:t,succeed:r}}var y=class r{static#r=new Map;#e;#t;constructor(e){this.#e=e,this.#t=null}static async get(e){let t=`${e.schemaHash}:${e.clientVersion}`;try{let n=r.#r.get(t);if(n)return n;let o=new r(e);return r.#r.set(t,o),e.debug&&console.debug("starting engine...",e),await o.start(),e.debug&&console.debug("engine started!"),o}finally{r.stopAll(t)}}static async stopAll(e){let n=(await Promise.allSettled(Array.from(r.#r.entries()).filter(([o])=>o!==e).map(async([o,s])=>{try{await s.stop()}finally{r.#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,t){return await this.#i(e,t,"commit")}async request(e,t){let{url:n}=await this.#t,o=this.#n(t),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,t){return await this.#i(e,t,"rollback")}async startTransaction(e,t){let{url:n}=await this.#t,o=this.#n(t),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 e=await this.#o();this.#e.debug&&console.debug("spinning up engine at path...",e);let t=Me(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});t.stderr.setEncoding("utf8"),t.stdout.setEncoding("utf8");let{fail:n,promise:o,succeed:s}=T();this.#t=o;let a=i=>{let c=i.split(`
|
|
3
|
+
`).find(u=>u.includes("Started query engine http server"));if(!c)return;t.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:b,port:m}=l;if(b==null||m==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:t,url:`http://${b}:${m}`})};t.once("error",i=>{this.#t=null,n(new w(String(i)))}),t.once("exit",(i,c)=>{this.#t=null,n(new w(`Query Engine exited with code ${i} and signal ${c}`))}),t.stdout.on("data",a),t.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 je(e,"exit"))}async#o(){this.#e.debug&&console.debug("getting engine commit hash...");let e=await this.#s();this.#e.debug&&console.debug("got engine commit hash",e);let t=K(this.#e.clientVersion,e);this.#e.debug&&console.debug("cache directory path",t),await Fe(t,{recursive:!0});let{platform:n}=this.#e.platform,o=n==="windows"?".exe":"",s=Ue(t,`query-engine-${n}${o}`);return this.#e.debug&&console.debug("engine binary path",s),(Ne.env.PRISMA_DEV_FORCE_ENGINE_DOWNLOAD==="1"||await D(s)===!1)&&await this.#a({commitHash:e,extension:o,engineBinaryPath:s}),s}async#s(){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:t,extension:n,engineBinaryPath:o}=e,{binaryTarget:s}=this.#e.platform,a=`https://binaries.prisma.sh/all_commits/${t}/${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 z(await i.arrayBuffer(),o),this.#e.debug&&console.debug("engine downloaded and saved at",o)}#n(e){let t={};for(let[n,o]of Object.entries(e))o!=null&&(t[n]=o);return t}async#i(e,t,n){let{url:o}=await this.#t,s=this.#n(t),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 R(r,e){return console.error(r),r instanceof w?e.json({EngineNotStarted:{reason:{EngineStartupError:{logs:[],msg:r.message}}}},500):r instanceof g?e.text(r.responseBody,r.statusCode):e.body(null,500)}var w=class extends Error{name="EngineStartError"},g=class r extends Error{constructor(t,n,o){super(`${t}: Query Engine response status ${n}, body: ${o}`);this.action=t;this.statusCode=n;this.responseBody=o}name="EngineHttpError";static async fromResponse(t){let n=new URL(t.url),o=await t.text();return new r(n.pathname,t.status,o)}};var Q=51214,W=51213,X=51215,h=class extends Error{constructor(t,n){super(`Port number \`${t}\` is not available for service ${n}.`);this.port=t;this.service=n}name="PortNotAvailableError"};import{Buffer as Y}from"buffer";var O=new Map;async function k(r){let t=new TextEncoder().encode(r),n=await crypto.subtle.digest("SHA-256",t);return Array.from(new Uint8Array(n)).map(a=>a.toString(16).padStart(2,"0")).join("")}function Z(r){let e=r.req.param("schemaHash"),t=O.get(e);return t==null?r.json({EngineNotStarted:{reason:"SchemaMissing"}},404):{schemaHash:e,schemas:t}}var qe=/datasource\s+db\s+\{\s*provider\s*=\s*"postgres(!?ql)?"\s+url\s*=\s*.+\s*\}/;async function ee(r,e){let t=Y.from(r,"base64").toString("utf8"),n=`datasource db {
|
|
5
5
|
provider = "postgresql"
|
|
6
6
|
url = "${e}"
|
|
7
|
-
}`,o=
|
|
8
|
-
`,{encoding:"utf-8"})}};async
|
|
7
|
+
}`,o=t.replace(qe,n),s=await k(o);return{base64Override:Y.from(o,"utf8").toString("base64"),overrideHash:s}}function x(r){let{req:e}=r;return{traceparent:e.header("traceparent"),"X-capture-telemetry":e.header("X-capture-telemetry")}}import{integer as te,looseObject as Ve,minValue as re,number as _,object as Ke,optional as Ge,pipe as ne,safeParse as oe,string as se,union as ze}from"valibot";var Je=Ke({isolation_level:Ge(se()),max_wait:ne(_(),te(),re(0)),timeout:ne(_(),te(),re(0))});async function ae(r){let{issues:e,output:t,success:n}=oe(Je,await r.req.json(),{abortEarly:!0});return n?t:r.json({EngineNotStarted:{reason:"InvalidRequest",issues:e}},400)}var Qe=Ve({id:ze([se(),_()])});function ie(r,e){let{output:t,success:n}=oe(Qe,r);return n?t:e.json({EngineMalfunction:{}},500)}async function ue(r,e){let{port:t}=e;if(e.dryRun)return ce(t,null);let n=await rt(t,r,e),{fail:o,promise:s,succeed:a}=T(),i=Ye({createServer:We,fetch:n.fetch,port:t},()=>{a()});return i.on("error",c=>{if(typeof c=="object"&&"code"in c&&c.code==="EADDRINUSE")return o(new h(t,"server"));console.error(c)}),await s,ce(t,i)}function ce(r,e){return{async close(){e&&await Promise.allSettled([Xe(e.close.bind(e))(),y.stopAll()])},port:r,url:`http://localhost:${r}`}}async function rt(r,e,t){let{debug:n}=t,o=new de,s=await Ze.getPlatformInfo();return n&&console.debug("platform info: %s",JSON.stringify(s)),n&&o.use("*",et()),o.use("*",async(a,i)=>(a.set("db",e),a.set("debug",!!n),a.set("platform",s),a.set("port",r),a.set("protocol","http"),a.set("serverState",t),await i())),o.route("/",P),o}var P=new de;P.get("/health",r=>r.json({name:r.get("serverState").name}));P.post("/invalidate",S("header",p),async r=>{let e=await q(r);return e instanceof Response?e:r.body(null)});var nt="/:clientVersion/:schemaHash",v=P.basePath(nt);P.route("/",v);var ot=["/graphql","/itx/:transactionId/graphql"];v.on("POST",[...ot],S("header",p),async r=>{let{req:e}=r;try{let t=await H(r);if(t instanceof Response)return t;let n=await e.text(),o=e.param("transactionId"),s=await t.request(n,{...x(r),"X-transaction-id":o});return r.text(s)}catch(t){return R(t,r)}});v.basePath("/itx/:transactionId").on("POST",["/commit","/rollback"],S("header",p),async r=>{let{req:e}=r;try{let t=await H(r);if(t instanceof Response)return t;let o=`${e.routePath.split("/").filter(Boolean).at(-1)}Transaction`,s=e.param("transactionId"),a=await t[o](s,x(r));return r.json(a)}catch(t){return R(t,r)}});v.put("/schema",S("header",p),async r=>{let{req:e}=r,t=await e.text();if(!t)return r.text("Missing schema",400);let n=e.param("schemaHash"),o=O.get(n);if(o==null){if(n!==await k(t))return r.text("Schema hash mismatch",400);let s=await ee(t,r.get("db").connectionString);return O.set(n,{base64Original:t,...s}),r.text(n)}return t!==o.base64Original?r.text("Schema mismatch",400):r.text(n)});v.post("/transaction/start",S("header",p),async r=>{let{req:e}=r,t=await ae(r);if(t instanceof Response)return t;try{let n=await H(r);if(n instanceof Response)return n;let o=await n.startTransaction(t,x(r)),s=ie(o,r);if(s instanceof Response)return s;let{id:a}=s,i=e.param("clientVersion"),c=r.get("port"),l=r.get("protocol"),b=e.param("schemaHash");return r.json({...o,"data-proxy":{endpoint:`${l}://localhost:${c}/${i}/${b}/itx/${a}`}})}catch(n){return R(n,r)}});async function H(r){let{req:e}=r,t=Z(r);if(t instanceof Response)return t;let{base64Override:n,overrideHash:o}=t.schemas;return await y.get({base64Schema:n,clientVersion:tt.env.PRISMA_DEV_FORCE_CLIENT_VERSION||e.param("clientVersion"),debug:r.get("debug"),platform:r.get("platform"),schemaHash:o})}import{PGlite as st}from"@electric-sql/pglite";import{PGLiteSocketServer as at}from"@electric-sql/pglite-socket";import{pgDump as it}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"},ct=`postgres://${d.username}:${d.password}@localhost`,dt=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 B(r,e){let t=r==="database"?e.databasePort:e.shadowDatabasePort;if(e.dryRun)return le(r,e,{db:null,port:t,server:null});let{debug:n}=e,o=await st.create({database:d.database,dataDir:"memory://",debug:n?5:void 0,defaultDataTransferContainer:"file",username:d.username});await o.waitReady,r!=="shadow_database"&&await ut(o,e);let s=new at({db:o,inspect:n,port:t});try{await s.start()}catch(a){throw a instanceof Error&&"code"in a&&a.code==="EADDRINUSE"?new h(t,r):a}return le(r,e,{db:o,port:t,server:s})}function le(r,e,t){let{debug:n}=e,{db:o,port:s,server:a}=t||{};return n&&console.debug(`${r} server started on port %i`,s),{...d,close:async()=>{let i=[];try{await a?.stop(),n&&console.debug(`${r} server stopped on port %i`,s)}catch(c){console.error("server stop error",c),i.push(c)}if(r!=="shadow_database")try{await lt(o,e),n&&console.debug(`${r} state saved`)}catch(c){console.error("dump error",c),i.push(c)}try{await o?.close(),n&&console.debug(`${r} closed`)}catch(c){console.error("db close error",c),i.push(c)}if(i.length>0)throw new AggregateError(i,`Failed to close ${r} properly`)},connectionString:mt(s),port:s}}async function ut(r,e){try{let t=await e.readDatabaseDump();if(!t.length)return;await r.exec(`${t}; set search_path to public;`)}catch(t){if(t instanceof Error&&"code"in t&&t.code==="ENOENT")return console.warn("No database dump found. Starting with an empty database.");throw t}}async function lt(r,e){await e.writeDatabaseDump(()=>it({args:[],pg:r}))}function mt(r){return`${ct}:${r}/${d.database}?${dt.toString()}`}import{copyFile as pt,mkdir as gt,readFile as ht,unlink as ft,writeFile as me}from"fs/promises";import{join as pe}from"pathe";import{lock as bt}from"proper-lockfile";import{read as yt}from"read-last-lines";import{process as wt}from"std-env";var M=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 t=e?.dryRun!==!0&&e?.persistenceMode!=="stateless"?new L(e):new C(e);return await t[M](),t}},C=class extends f{constructor(e){super({...e,persistenceMode:"stateless"})}async[M](){}async close(){}readDatabaseDump(){return Promise.resolve("")}async writeDatabaseDump(){}async writeServerDump(){}},L=class extends f{#r;#e;#t;#o;#s=null;constructor(e){super({...e,persistenceMode:"stateful"}),this.#r=G(this.name),this.#e=pe(this.#r,"ppg.sql"),this.#t=`${this.#e}.bak`,this.#o=pe(this.#r,"prisma-dev.json")}async[M](){await gt(this.#r,{recursive:!0}),this.debug&&console.debug(`using data directory: ${this.#r}`);try{this.#s=await bt(this.#r,{lockfilePath:`${this.#o.replace("json","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.#s!=null&&(await this.#s(),this.debug&&console.debug(`released lock on: ${this.#r}`))}async readDatabaseDump(){let e=await this.#a(this.#e);if(e){let n=await A(this.#t);return this.debug&&n&&console.debug(`deleted backup database dump: ${this.#t}`),e}let t=await this.#a(this.#t);return t&&(await me(this.#e,t,{encoding:"utf-8"}),await ft(this.#t),this.debug&&console.debug(`restored dump from backup: ${this.#t}`)),t||""}async writeDatabaseDump(e){let t=!1;if(await this.#n(this.#e))try{await pt(this.#e,this.#t),t=!0,this.debug&&console.debug(`backed up dump to: ${this.#t}`)}catch(s){if(!(s instanceof Error&&"code"in s&&s.code==="ENOENT"))throw s}let n=await e();if(await J(n,this.#e),this.debug&&console.debug(`dumped database to: ${this.#e}`),!t)return;let o=await A(this.#t);this.debug&&o&&console.debug(`deleted backup dump: ${this.#t}`)}async writeServerDump(e){await me(this.#o,`${JSON.stringify({name:this.name,pid:wt.pid,port:this.port,databasePort:this.databasePort,shadowDatabasePort:this.shadowDatabasePort,exports:e},null,2)}
|
|
8
|
+
`,{encoding:"utf-8"})}async#a(e){if(!await this.#n(e)){let n=await A(e);return this.debug&&n&&console.debug(`deleted incomplete database dump at: ${e}`),null}let t=await ht(e,{encoding:"utf-8"});return this.debug&&console.debug(`read database dump from: ${e}`),t}async#n(e){if(!await D(e))return!1;let t=await yt(e,5,"utf-8");return this.debug&&console.debug(`last lines of checked dump: ${t}`),t.includes("-- PostgreSQL database dump complete")}};async function Rr(r){let e=await f.get(r),[t,n]=await Promise.all([B("database",e),B("shadow_database",e)]),o=await ue(t,e),s=`prisma+postgres://localhost:${o.port}/?${new URLSearchParams({api_key:N({databaseUrl:t.connectionString,shadowDatabaseUrl:n.connectionString})}).toString()}`,a={accelerate:{url:o.url},database:{connectionString:t.connectionString},ppg:{url:s},shadowDatabase:{connectionString:n.connectionString}};return await e.writeServerDump(a),{...a,close:()=>i(e,[o,t,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{Q as DEFAULT_DATABASE_PORT,W as DEFAULT_SERVER_PORT,X as DEFAULT_SHADOW_DATABASE_PORT,h as PortNotAvailableError,Rr 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.202505221234",
|
|
4
4
|
"description": "A local Prisma Postgres server for development and testing",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "./dist/index.js",
|
|
@@ -47,6 +47,7 @@
|
|
|
47
47
|
"pako": "2.1.0",
|
|
48
48
|
"pathe": "2.0.3",
|
|
49
49
|
"proper-lockfile": "4.1.2",
|
|
50
|
+
"read-last-lines": "1.8.0",
|
|
50
51
|
"std-env": "3.8.1",
|
|
51
52
|
"valibot": "1.1.0"
|
|
52
53
|
},
|