UNPKG

101 kBJavaScriptView Raw
1#!/usr/bin/env node
2var tt=Object.create;var O=Object.defineProperty,rt=Object.defineProperties,ot=Object.getOwnPropertyDescriptor,st=Object.getOwnPropertyDescriptors,it=Object.getOwnPropertyNames,le=Object.getOwnPropertySymbols,nt=Object.getPrototypeOf,ce=Object.prototype.hasOwnProperty,at=Object.prototype.propertyIsEnumerable;var pe=(e,t,o)=>t in e?O(e,t,{enumerable:!0,configurable:!0,writable:!0,value:o}):e[t]=o,b=(e,t)=>{for(var o in t||(t={}))ce.call(t,o)&&pe(e,o,t[o]);if(le)for(var o of le(t))at.call(t,o)&&pe(e,o,t[o]);return e},k=(e,t)=>rt(e,st(t)),ue=e=>O(e,"__esModule",{value:!0});var me=(e,t)=>{ue(e);for(var o in t)O(e,o,{get:t[o],enumerable:!0})},lt=(e,t,o)=>{if(t&&typeof t=="object"||typeof t=="function")for(let r of it(t))!ce.call(e,r)&&r!=="default"&&O(e,r,{get:()=>t[r],enumerable:!(o=ot(t,r))||o.enumerable});return e},c=e=>lt(ue(O(e!=null?tt(nt(e)):{},"default",e&&e.__esModule&&"default"in e?{get:()=>e.default,enumerable:!0}:{value:e,enumerable:!0})),e);me(exports,{registerRuntime:()=>W});var Ye=c(require("path")),et=c(require("sade"));var ct="presta",pt="0.43.9",ut="Hyper minimal framework for the modern web.",mt="./dist/index.js",ft="./dist/index.d.ts",dt={presta:"dist/bin.js"},gt=["dist"],ht={build:"node scripts/build && pnpx tsc --emitDeclarationOnly",cloc:"pnpm dlx cloc './lib/*.ts'",typecheck:"pnpx tsc --noEmit"},yt={type:"git",url:"git+ssh://git@github.com/sure-thing/presta.git"},bt="estrattonbailey",vt="MIT",xt={url:"https://github.com/sure-thing/presta/issues"},wt="https://github.com/sure-thing/presta#readme",Pt={chokidar:"^3.4.3","deep-extend":"^0.6.0",dotenv:"^10.0.0",esbuild:"^0.12.28","esbuild-register":"^2.6.0",filewatcher:"^3.0.1","fs-extra":"^9.0.1","get-port":"^5.1.1",kleur:"^4.1.4","mime-types":"^2.1.31","module-alias":"^2.2.2",ms:"^2.1.2",picomatch:"^2.3.0","query-string":"^6.14.1","raw-body":"^2.4.1",regexparam:"^1.3.0","route-sort":"^1.0.0",sade:"^1.7.3",sirv:"^1.0.7",statuses:"^2.0.1","tiny-glob":"^0.2.9","watch-dependency-graph":"^3.0.1",ws:"^8.4.0"},St={"@netlify/functions":"^0.7.2","@types/deep-extend":"^0.4.32","@types/fs-extra":"^9.0.12","@types/mime-types":"^2.1.0","@types/picomatch":"^2.2.4","@types/sade":"^1.7.3","@types/statuses":"^2.0.0","@types/ws":"^8.2.2",proxyquire:"^2.1.3","supertest-fetch":"^1.4.3",typescript:"^4.5.2"},fe={name:ct,version:pt,description:ut,main:mt,types:ft,bin:dt,files:gt,scripts:ht,repository:yt,author:bt,license:vt,bugs:xt,homepage:wt,dependencies:Pt,devDependencies:St};var Ge=c(require("fs-extra")),Je=c(require("path")),ae=c(require("chokidar"));var Z={};me(Z,{Levels:()=>w,colors:()=>D.default,debug:()=>h,error:()=>d,getLogs:()=>Ct,info:()=>y,logger:()=>F,newline:()=>kt,raw:()=>Rt,warn:()=>C});var D=c(require("kleur"));var v={PRODUCTION:"production",DEVELOPMENT:"development"};var w;(function(i){i.Debug="debug",i.Info="info",i.Warn="warn",i.Err="error"})(w||(w={}));var J=[],Et={[w.Debug]:"magenta",[w.Info]:"blue",[w.Warn]:"yellow",[w.Err]:"red"};function Ct(){if(!process.env.TESTING)throw new Error("Internal method was called outside test mode");return J}function F(e){if(process.env.TESTING)J.push(e);else{let t=process.env.PRESTA_DEBUG,o=process.env.PRESTA_ENV===v.PRODUCTION?"prod":"dev";if(!t&&e.level===w.Debug)return;console.log([D.default.gray(o),D.default[Et[e.level||"info"]](e.label),e.message,e.duration?D.default.gray("+"+e.duration):"",e.error?`
3
4${e.error.stack||e.error}
5
6`:""].filter(Boolean).join(" "))}}function h(e){F(b({level:w.Debug},e))}function y(e){F(b({level:w.Info},e))}function C(e){F(b({level:w.Warn},e))}function d(e){F(b({level:w.Err},e))}function Rt(...e){process.env.TESTING?J.push(e):console.log(...e)}function kt(){process.env.TESTING||console.log("")}var de=c(require("fs")),S=c(require("path")),ge=c(require("get-port"));var E="presta.config.js";function T(e){return(0,ge.default)({port:parseInt(e,10)})}function M(e,t=!1){let o=S.default.resolve(e||E);try{return delete require.cache[o],require(o)}catch(r){return de.default.existsSync(o)&&(d({label:"error",error:r}),t&&process.exit(1)),{}}}function j(e,t,o){let r=b({env:e,output:"build",assets:"public",plugins:[],port:4e3,files:[]},o);return t._.length&&(r.files=t._),t.output&&(r.output=t.output),t.assets&&(r.output=t.assets),t.port&&(r.port=t.port),r.files&&(r.files=[].concat(r.files).map(i=>S.default.resolve(process.cwd(),i))),r.output&&(r.output=S.default.resolve(process.cwd(),r.output)),r.assets&&(r.assets=S.default.resolve(process.cwd(),r.assets)),k(b({},r),{staticOutputDir:S.default.join(r.output,"static"),functionsOutputDir:S.default.join(r.output,"functions"),functionsManifest:S.default.join(r.output,"routes.json")})}var Y=c(require("fs-extra")),Ee=c(require("path")),Ce=c(require("watch-dependency-graph")),Re=c(require("chokidar")),ke=c(require("picomatch")),Oe=c(require("deep-extend"));var L=c(require("fs-extra")),ye=c(require("path")),be=c(require("route-sort"));function he(e){for(var t=5381,o=e.length;o;)t=t*33^e.charCodeAt(--o);return(t>>>0).toString(36)}function Ot(e){return e.replace(process.cwd(),"").split(".").reverse().slice(1).reverse().join("-").split("/").filter(Boolean).join("-")}function $(e,t){let o=e.map(s=>{try{let{route:n}=require(s),p=Ot(s),a=ye.default.join(t.functionsOutputDir,t.env===v.PRODUCTION?p+"-"+he(L.default.readFileSync(s,"utf8"))+".js":p+".js");return h({label:"debug",message:`generating ${p} lambda`}),delete require.cache[s],delete require.cache[a],L.default.outputFileSync(a,`import { wrapHandler } from 'presta/dist/wrapHandler';
7 import * as file from '${s}';
8 export const route = file.route
9 export const handler = wrapHandler(file)`),[n,a]}catch(n){d({label:"error",error:n})}}).filter(Boolean),r=(0,be.default)(o.map(s=>s[0])),i={};for(let s of r){let n=o.find(p=>p[0]===s);n&&(i[s]=n[1])}return L.default.outputFileSync(t.functionsManifest,JSON.stringify(i)),o}var K=c(require("fs-extra")),ve=c(require("path")),xe=c(require("tiny-glob/sync"));function H(e){return/export\s.+\sroute\s+\=/.test(K.default.readFileSync(e,"utf-8"))}function q(e){return/export\s.+\sgetStaticPaths/.test(K.default.readFileSync(e,"utf-8"))}function A(e){try{return[].concat(e).map(t=>(0,xe.default)(t)).flat().map(t=>ve.default.resolve(process.cwd(),t))}catch(t){return d({label:"paths",message:"no files found",error:t}),[]}}var X=c(require("fs-extra")),B=c(require("path")),Se=c(require("mime-types"));function P(){let e=process.hrtime();return()=>{let[t,o]=process.hrtime(e),r=o/1e6;return t<1?(r>=1?r.toFixed(0):r.toFixed(2))+"ms":t+"."+r.toFixed(0)+"s"}}var we=c(require("regexparam"));function Pe(e,t){let[o]=e.split("?"),r=(0,we.default)(t),i=0,s={},n=r.pattern.exec(o)||[];for(;i<r.keys.length;)s[r.keys[i]]=n[++i];return s}function Dt(e){return typeof e=="object"?JSON.stringify(e):e}function R(e){let{isBase64Encoded:t=!1,statusCode:o=200,headers:r={},multiValueHeaders:i={},body:s="",html:n=void 0,json:p=void 0,xml:a=void 0}=typeof e=="string"?{body:e}:e,l="text/html; charset=utf-8";return p?l="application/json; charset=utf-8":a&&(l="application/xml; charset=utf-8"),{isBase64Encoded:t,statusCode:o,headers:b({"Content-Type":l},r),multiValueHeaders:i,body:Dt(s||n||p||a||"")}}function N({port:e}){return`
10 <script>
11 (function (global) {
12 var socket = new WebSocket('ws://localhost:${e}');
13
14 socket.addEventListener('open', function (event) {
15 console.log('[presta] connected on port ${e}')
16 });
17
18 socket.addEventListener('message', function (event) {
19 console.log(\`'[presta] received \${event.data}\`)
20 if (event.data === 'refresh') {
21 global.location.reload();
22 }
23 });
24
25 socket.addEventListener('close', function () {
26 console.log('[presta] disconnected')
27 });
28 })(this);
29 <\/script>
30 `}function Ft(e,t="html"){return B.default.extname(e)?e:t==="html"?`${e}/index.html`:`${e}.${t}`}async function V(e){return h({label:"debug",message:`removing old static file ${e}`}),X.default.remove(e)}async function Ht(e){return Promise.all(e.map(V))}async function Bt(e,t,{footer:o}){let r=require(e),i=await r.getStaticPaths(),s=[];if(!i||!i.length)return s;for(let n of i){let p=P(),a={path:n,pathParameters:r.route?Pe(n,r.route):{}},l=R(await r.handler(a,{})),u=l.headers?l.headers["Content-Type"]:"",m=u&&Se.default.extension(u)||"html",g=Ft(n,m),x=l.body+o;X.default.outputFileSync(B.default.join(t,g),x,"utf-8"),y({label:"built",message:n,duration:p()}),s.push(g)}return s}async function _(e,t,o={}){let r=t.env===v.DEVELOPMENT,i=t.staticOutputDir,s=r?N({port:t.port}):"";for(let n of e)try{let p=n.replace(process.cwd(),""),a=o[n]||[],l=await Bt(n,i,{footer:s});if(!l||!l.length){C({label:"paths",message:`${p} - no paths to render`}),Ht(a.map(u=>B.default.join(i,u)));continue}for(let u of a)l.includes(u)||V(B.default.join(i,u));o[n]=l}catch(p){d({label:"error",error:p});break}return{staticFilesMap:o}}function De(e,t){let o=P();$(e,t),e.length&&y({label:"built",message:"lambdas",duration:o()})}function It(e,t,o){return(0,ke.default)(t)(e)&&!o.includes(e)}async function Fe(e,t){let o={},r=A(e.files);r.length||C({label:"paths",message:"no files configured"});async function i(a,l,u){if(delete require.cache[a],q(a)){let m=await _([a],u,o);o=(0,Oe.default)({},o,m.staticFilesMap)}De(l.filter(H),u)}async function s(a,l,u){for(let m of a)await i(m,l,u)}await s(r,r,e),t.emitBrowserRefresh();let n=(0,Ce.create)({alias:{"@":process.cwd()}});n.onChange(async a=>{await s(a,r,e),t.emitBrowserRefresh()}),n.onRemove(async([a])=>{h({label:"watch",message:`removed ${a}`}),r.splice(r.indexOf(a),1),De(r.filter(H),e),(o[a]||[]).forEach(l=>V(Ee.default.join(e.staticOutputDir,l))),t.emitBrowserRefresh()}),n.onError(a=>{d({label:"error",error:typeof a=="string"?new Error(a):a})}),await n.add(r);let p=Re.default.watch(process.cwd(),{ignoreInitial:!0,ignored:[e.output,e.assets]});return p.on("add",async a=>{!Y.default.existsSync(a)||Y.default.lstatSync(a).isDirectory()||!It(a,e.files,r)||(h({label:"watch",message:`add ${a}`}),r.push(a),await n.add(a),await i(a,r,e),t.emitBrowserRefresh())}),t.onBuildFile(async({file:a})=>{await i(a,r,e),t.emitBrowserRefresh()}),{async close(){await n.close(),await p.close()}}}async function U(e,t,o){let r=await Promise.all(e.map(i=>{try{return i(t,o)}catch(s){d({label:"error",error:s})}}).filter(Boolean));return{async cleanup(){return Promise.all(r.map(i=>i&&i.cleanup&&i.cleanup()))}}}var He;(function(r){r.PostBuild="post-build",r.BuildFile="build-file",r.BrowserRefresh="browser-refresh"})(He||(He={}));function z(){let e={};function t(s,...n){e[s]&&e[s].map(p=>p(...n))}function o(s,n){return e[s]=e[s]?e[s].concat(n):[n],()=>e[s].splice(e[s].indexOf(n),1)}function r(){e={}}function i(s){return e[s]||[]}return{emit:t,on:o,clear:r,listeners:i}}function Q(e){return{emitPostBuild(t){e.emit("postBuild",t)},onPostBuild(t){return e.on("postBuild",t)},emitBuildFile(t){e.emit("buildFile",t)},onBuildFile(t){return e.on("buildFile",t)},emitBrowserRefresh(){e.emit("browserRefresh")},onBrowserRefresh(t){return e.on("browserRefresh",t)}}}var Be=c(require("path")),re=c(require("fs-extra")),Ie=c(require("esbuild"));function ee(e){return delete require.cache[e],require(e)}function te(e){try{return ee(e)}catch(t){return{}}}async function Te(e,t){let o=P(),r=A(e.files),i=r.filter(q),s=r.filter(H);if(h({label:"build",message:"starting build"}),!i.length&&!s.length)C({label:"files",message:"no files were found, nothing to build"});else{let n="",p=0,a="",l="",u=await Promise.allSettled([(async()=>{if(i.length){let m=P(),{staticFilesMap:g}=await _(i,e);n=m(),p=Object.keys(g).reduce((x,G)=>x+=g[G].length,0)}})(),(async()=>{if(s.length){let m=P(),g=te(Be.default.join(process.cwd(),"package.json"));$(s,e),await(0,Ie.build)({entryPoints:Object.values(require(e.functionsManifest)),outdir:e.functionsOutputDir,platform:"node",target:["node12"],minify:!0,allowOverwrite:!0,external:Object.keys(g.dependencies||{}),bundle:!0,define:{"process.env.PRESTA_SERVERLESS_RUNTIME":"true"}}),a=m()}})(),(async()=>{if(re.default.existsSync(e.assets)){let m=P();re.default.copySync(e.assets,e.staticOutputDir),l=m()}})()]);if(u.find(m=>m.status==="rejected"))throw h({label:"build",message:"build partially failed"}),u.forEach(m=>{m.status==="rejected"&&d({label:"error",error:m.reason})}),new Error("presta build failed");n&&y({label:"static",message:`rendered ${p} file(s)`,duration:n}),a&&y({label:"lambda",message:`compiled ${s.length} function(s)`,duration:a}),l&&y({label:"assets",message:`copied in ${l}`}),t.emitPostBuild({output:e.output,staticOutput:e.staticOutputDir,functionsOutput:e.functionsOutputDir,functionsManifest:te(e.functionsManifest)}),(n||a)&&y({label:"complete",message:`in ${o()}`})}}var Ue=c(require("http")),se=c(require("sirv")),ze=c(require("mime-types")),Qe=c(require("regexparam")),ie=c(require("statuses")),We=c(require("ws"));var oe=c(require("statuses"));function Me({statusCode:e}){return`<!-- built with presta https://npm.im/presta -->
31 <!DOCTYPE html>
32 <html>
33 <head>
34 <meta charset="UTF-8" />
35 <meta name="viewport" content="width=device-width,initial-scale=1" />
36 <title>${e} \u2014\xA0${oe.default.message[e]}</title>
37 <link rel="icon" type="image/png" href="https://presta.run/favicon.png">
38 <link rel="icon" type="image/svg" href="https://presta.run/favicon.svg">
39 <link rel="preconnect" href="https://fonts.googleapis.com">
40 <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
41 <link href="https://fonts.googleapis.com/css2?family=Inter:wght@400;900&display=swap" rel="stylesheet">
42 <link rel='stylesheet' href='https://unpkg.com/svbstrate@5.1.0/svbstrate.css' />
43 <style>
44 html,body {
45 font-family: 'Inter', 'sans-serif';
46 color: #23283D;
47 background-color: #DADEF0;
48 }
49 #favicon {
50 fill: #23283D;
51 }
52 @media (prefers-color-scheme: dark) {
53 html,body {
54 color: #DADEF0;
55 background-color: #23283D;
56 }
57 #favicon {
58 fill: #DADEF0;
59 }
60 }
61 </style>
62 </head>
63 <body class='w f aic jcc' style='height: 100vh'>
64 <div class='p12 tac'>
65 <h1>${e}</h1>
66 <p class='mb1'>${oe.default.message[e]}</p>
67
68 <svg width="32" height="32" viewBox="0 0 32 32" fill="none" xmlns="http://www.w3.org/2000/svg">
69 <g clip-path="url(#a)">
70 <path id="favicon" fill-rule="evenodd" clip-rule="evenodd" d="M10.4 7c-.3 0-.8.2-1 .5L1.1 22.1c-.2.3 0 .6.3.6l4 .3-2.1 2.6c-.2.3-.1.6.2.6l16.8 1.3c.4 0 .8-.2 1-.4L32 13.9c.2-.2.1-.5-.2-.5l-6.4-.5 2.2-4c.2-.3 0-.5-.3-.6L10.4 7ZM24 12.8l1.9-3.4-15.5-1.2-7.7 13.4 3.6.3 7.5-9.4c.3-.3.7-.5 1-.4l9.2.7ZM7.6 22l7.1-8.9 8.7.7-5.2 9L7.6 22Zm-1 1.1 11.6 1c.3 0 .8-.3 1-.6l5.5-9.6 5.5.5-9.7 12L5 25.2l1.7-2Z" fill="#23283D"/>
71 </g>
72 <defs>
73 <clipPath id="a">
74 <path fill="#fff" d="M0 0h32v32H0z"/>
75 </clipPath>
76 </defs>
77 </svg>
78 </div>
79 </body>
80 </html>
81 `}var qe=c(require("url")),Ae=c(require("raw-body")),Ne=c(require("mime-types"));function je(e){let t={},o={};for(let r of Object.keys(e)){let i=r.toLowerCase(),s=e[r];!s||(Array.isArray(s)?o[i]=s:t[i]=s)}return{headers:t,multiValueHeaders:o}}var Le=c(require("query-string"));function $e(e){let t=(0,Le.parse)(e,{arrayFormat:"comma"}),o={},r={};for(let i of Object.keys(t)){let s=t[i];Array.isArray(s)?r[i]=s:s&&(o[i]=s)}return{queryStringParameters:o,multiValueQueryStringParameters:r}}var Tt=/image|audio|video|application\/pdf|application\/zip|applicaton\/octet-stream/i;function Mt(e){return Boolean(e)&&Tt.test(e)}async function Ve(e){let{url:t="",method:o}=e,{headers:r,multiValueHeaders:i}=je(e.headers),s=Mt(r["content-type"]||""),p=r["content-length"]?await(0,Ae.default)(e,{limit:"1mb",encoding:r["content-type"]&&Ne.default.charset(r["content-type"])||!0}):void 0,a=(0,qe.parse)(t).query||"",{queryStringParameters:l,multiValueQueryStringParameters:u}=$e(a);return{rawUrl:t,path:t,httpMethod:o,headers:r,multiValueHeaders:i,rawQuery:a,queryStringParameters:l,multiValueQueryStringParameters:u,body:p?Buffer.from(p).toString(s?"base64":"utf8"):null,isBase64Encoded:s,pathParameters:{}}}function _e(e,t){let o=R(t);for(let r in t.multiValueHeaders)e.setHeader(r,String(t.multiValueHeaders[r]));for(let r in t.headers)e.setHeader(r,String(t.headers[r]));e.statusCode=o.statusCode,e.write(o.body),e.end()}function jt(e,t){let o=new Error(t);return o.statusCode=e,o}function Lt(e){let t=((e==null?void 0:e.headers)||{})["Content-Type"]||"html";return t&&ze.default.extension(String(t))||"html"}function $t(e,t){let r=Object.keys(t).map(i=>({matcher:(0,Qe.default)(i),route:i})).filter(({matcher:i})=>i.pattern.test(e.split("?")[0])).map(({route:i})=>t[i])[0];return r?require(r):void 0}async function qt(e,t){let o=e.headers.Accept||e.headers.accept,r=o&&o.includes("json");try{if(!t||!t.handler)throw jt(404,"");return R(await t.handler(e,{awsRequestId:"presta dev"}))}catch(i){let s=i,{statusCode:n=500}=s;return n>499&&d({label:"error",message:s.message||ie.default.message[n],error:s}),R({statusCode:n,html:r?void 0:Me({statusCode:n}),json:r?{detail:ie.default.message[n]}:void 0})}}function At({port:e,config:t}){return async function(r,i){var g;let s=P(),n=await Ve(r),p=ee(t.functionsManifest),a=$t(n.path,p),l=await qt(n,a),u=l.statusCode>299&&l.statusCode<399;Lt(l)==="html"&&(l.body=(l.body||"").split("</body>")[0]+N({port:e})),Z[l.statusCode<299?"info":"error"]({label:"serve",message:`${l.statusCode} ${u&&((g=l==null?void 0:l.headers)==null?void 0:g.Location)||n.path}`,duration:s()}),_e(i,l)}}function Nt({port:e,config:t}){let o=t.staticOutputDir,r=t.assets;return async function(s,n){let p=P(),a=s.url;h({label:"debug",message:`handling ${a}`});function l(u,m){y({label:"serve",message:`${u.statusCode} ${m}`,duration:p()})}(0,se.default)(r,{dev:!0,setHeaders:l})(s,n,()=>{(0,se.default)(o,{dev:!0,setHeaders:l})(s,n,async()=>{At({port:e,config:t})(s,n)})})}}function ne(e,t){let o=e.port,r=Ue.default.createServer(Nt({port:o,config:e})).listen(o),i=new We.WebSocketServer({server:r}),s=[];return r.on("connection",n=>{s.push(n),n.on("close",()=>s.splice(s.indexOf(n),1))}),t.onBrowserRefresh(()=>{h({label:"debug",message:"refresh event received"}),i.clients.forEach(n=>n.send("refresh"))}),{async close(){await new Promise(n=>{r.close(()=>n(1)),s.forEach(p=>p.destroy())})}}}async function Ze(e){let t=M(e.config,!0),o=await T(e.port||t.port||4e3),r=z(),i=Q(r),s=j(v.PRODUCTION,k(b({},e),{port:o}),t);await U(s.plugins,s,i),Ge.default.emptyDirSync(s.output),y({label:"build"}),await Te(s,i)}async function Ke(e){let t=e["no-serve"],o,r,i=!1;async function s(){let p,a,l,u=M(e.config);(!r||u.port&&r!==u.port)&&(r=await T(e.port||u.port||4e3),process.env.PRESTA_SERVE_URL=`http://localhost:${r}`);let m=z(),g=Q(m),x=j(v.DEVELOPMENT,k(b({},e),{port:r}),u),G=await U(x.plugins,x,g);return h({label:"debug",message:`config created ${JSON.stringify(x)}`}),t||(a=ne(x,g),l=ae.default.watch(x.assets,{ignoreInitial:!0}).on("all",()=>{g.emitBrowserRefresh()})),y({label:i?"restart":"start",message:t?"":`http://localhost:${x.port}`}),p=await Fe(x,g),{config:x,async close(){m.clear(),await G.cleanup(),await l.close(),await p.close(),a&&await a.close()}}}let n=ae.default.watch(Je.default.resolve(e.config||E),{ignoreInitial:!0}).on("all",async()=>{if(!i){i=!0;try{await o.close()}catch(p){console.error(p)}console.clear(),o=await s(),i=!1}});return o=await s(),{async close(){await n.close(),await o.close()}}}async function Xe(e){let t=M(e.config,!0),o=await T(e.port||t.port||4e3),r=z(),i=Q(r),s=j(v.PRODUCTION,k(b({},e),{port:o}),t);await U(s.plugins,s,i),ne(s,i),y({label:"serve",message:`http://localhost:${s.port}`})}function W(e={}){require("dotenv").config({path:Ye.default.join(process.cwd(),".env")}),require("module-alias").addAliases({"@":process.cwd(),"presta:internal":__dirname}),require("esbuild-register/dist/node").register(e)}var I=(0,et.default)("presta");I.version(fe.version).option("--config, -c",`Path to a config file. (default ${E})`).option("--output, -o","Specify output directory for built files. (default ./build)").option("--assets, -a","Specify static asset directory. (default ./public)").option("--debug, -d","Enable debug mode (prints more logs)").example("dev index.jsx -o dist").example("dev 'pages/*.tsx' -o static").example("'pages/*.tsx'").example("-c site.json").example("serve -p 8080");I.command("build","Build project to output directory.",{default:!0}).example("").example("files/**/*.js").example(`-c ${E}`).action(e=>{process.env.PRESTA_ENV=v.PRODUCTION,process.env.PRESTA_DEBUG=e.debug?"debug":"",console.clear(),W(),Ze(e)});I.command("dev","Start Presta dev server and watch files",{alias:"watch"}).option("--port, -p","Port to run the local server. (default 4000)").option("--no-serve, -n","Do not run local dev server. (default false)").describe("Watch project and build to output directory.").example("dev").example("dev ./files/**/*.js").example("dev ./files/**/*.js -o ./out").example(`dev -c ${E}`).action(e=>{process.env.PRESTA_ENV=v.DEVELOPMENT,process.env.PRESTA_DEBUG=e.debug?"debug":"",console.clear(),W(),Ke(e)});I.command("serve").option("--port, -p","Port to run the local server. (default 4000)").describe("Serve built files, lambdas, and static assets.").example("serve").example("serve -o ./out -p 8080").example(`serve -c ${E}`).action(async e=>{process.env.PRESTA_ENV=v.DEVELOPMENT,process.env.PRESTA_DEBUG=e.debug?"debug":"",console.clear(),W(),Xe(e)});I.parse(process.argv);0&&(module.exports={registerRuntime});
82//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../lib/bin.ts", "../lib/cli.ts", "../lib/log.ts", "../lib/constants.ts", "../lib/config.ts", "../lib/watch.ts", "../lib/outputLambdas.ts", "../lib/hashContent.ts", "../lib/getFiles.ts", "../lib/renderStaticEntries.ts", "../lib/timer.ts", "../lib/getRouteParams.ts", "../lib/normalizeResponse.ts", "../lib/liveReloadScript.ts", "../lib/plugins.ts", "../lib/createEmitter.ts", "../lib/build.ts", "../lib/utils.ts", "../lib/serve.ts", "../lib/createDefaultHtmlResponse.ts", "../lib/requestToEvent.ts", "../lib/normalizeHeaders.ts", "../lib/parseQueryStringParameters.ts", "../lib/sendServerlessResponse.ts"],
  "sourcesContent": ["#!/usr/bin/env node\n\nimport path from 'path'\nimport sade from 'sade'\n\nimport pkg from '../package.json'\n\nimport { buildCommand, devCommand, serveCommand } from './cli'\nimport { defaultConfigFilepath } from './config'\nimport { Env } from './constants'\n\nexport function registerRuntime(options = {}) {\n  require('dotenv').config({ path: path.join(process.cwd(), '.env') })\n\n  require('module-alias').addAliases({\n    '@': process.cwd(),\n    'presta:internal': __dirname, // TODO wherever this is running from\n  })\n\n  require('esbuild-register/dist/node').register(options)\n}\n\nconst program = sade('presta')\n\nprogram\n  .version(pkg.version)\n  // do not provide default config here\n  .option('--config, -c', `Path to a config file.  (default ${defaultConfigFilepath})`)\n  .option('--output, -o', `Specify output directory for built files.  (default ./build)`)\n  .option('--assets, -a', `Specify static asset directory.  (default ./public)`)\n  .option('--debug, -d', `Enable debug mode (prints more logs)`)\n  .example(`dev index.jsx -o dist`)\n  .example(`dev 'pages/*.tsx' -o static`)\n  .example(`'pages/*.tsx'`)\n  .example(`-c site.json`)\n  .example(`serve -p 8080`)\n\nprogram\n  .command('build', 'Build project to output directory.', { default: true })\n  .example(``)\n  .example(`files/**/*.js`)\n  .example(`-c ${defaultConfigFilepath}`)\n  .action((options) => {\n    process.env.PRESTA_ENV = Env.PRODUCTION\n    process.env.PRESTA_DEBUG = options.debug ? 'debug' : ''\n    console.clear()\n    registerRuntime()\n    buildCommand(options)\n  })\n\nprogram\n  .command('dev', 'Start Presta dev server and watch files', { alias: 'watch' })\n  .option('--port, -p', `Port to run the local server.  (default 4000)`)\n  .option('--no-serve, -n', `Do not run local dev server.  (default false)`)\n  .describe('Watch project and build to output directory.')\n  .example(`dev`)\n  .example(`dev ./files/**/*.js`)\n  .example(`dev ./files/**/*.js -o ./out`)\n  .example(`dev -c ${defaultConfigFilepath}`)\n  .action((options) => {\n    process.env.PRESTA_ENV = Env.DEVELOPMENT\n    process.env.PRESTA_DEBUG = options.debug ? 'debug' : ''\n    console.clear()\n    registerRuntime()\n    devCommand(options)\n  })\n\nprogram\n  .command('serve')\n  .option('--port, -p', `Port to run the local server.  (default 4000)`)\n  .describe('Serve built files, lambdas, and static assets.')\n  .example(`serve`)\n  .example(`serve -o ./out -p 8080`)\n  .example(`serve -c ${defaultConfigFilepath}`)\n  .action(async (options) => {\n    process.env.PRESTA_ENV = Env.DEVELOPMENT\n    process.env.PRESTA_DEBUG = options.debug ? 'debug' : ''\n    console.clear()\n    registerRuntime()\n    serveCommand(options)\n  })\n\nprogram.parse(process.argv)\n", "import fs from 'fs-extra'\nimport path from 'path'\nimport chokidar from 'chokidar'\n\nimport * as logger from './log'\nimport { Options, create, getConfigFile, defaultConfigFilepath, getAvailablePort } from './config'\nimport { watch } from './watch'\nimport { initPlugins } from './plugins'\nimport { createEmitter, createHooks } from './createEmitter'\nimport { build } from './build'\nimport { serve } from './serve'\nimport { Env } from './constants'\n\nexport type PrestaCLIOptions = {\n  config?: string\n  output?: string\n  debug?: boolean\n} & Partial<Options>\n\nexport type PrestaCLIBuildOptions = PrestaCLIOptions & {\n  _: string[]\n}\n\nexport type PrestaCLIServeOptions = PrestaCLIBuildOptions & {\n  port?: string\n  'no-serve': boolean\n}\n\nexport type PrestaCLIDevOptions = PrestaCLIServeOptions\n\nexport async function buildCommand(options: PrestaCLIBuildOptions) {\n  const configFile = getConfigFile(options.config, true)\n  const port = await getAvailablePort(options.port || configFile.port || 4000)\n\n  const emitter = createEmitter()\n  const hooks = createHooks(emitter)\n  const config = create(Env.PRODUCTION, { ...options, port }, configFile)\n  await initPlugins(config.plugins, config, hooks)\n\n  fs.emptyDirSync(config.output)\n\n  logger.info({\n    label: 'build',\n  })\n\n  await build(config, hooks)\n}\n\nexport async function devCommand(options: PrestaCLIDevOptions) {\n  const noServe = options['no-serve']\n  let devServer: any\n  let port: number\n  let restarting = false\n\n  async function startDevServer() {\n    let watchTask: any\n    let httpServer: ReturnType<typeof serve>\n    let staticAssetWatcher: ReturnType<typeof chokidar.watch>\n\n    const userConfigFile = getConfigFile(options.config)\n\n    if (!port || (userConfigFile.port && port !== userConfigFile.port)) {\n      port = await getAvailablePort(options.port || userConfigFile.port || 4000)\n      process.env.PRESTA_SERVE_URL = `http://localhost:${port}`\n    }\n\n    const emitter = createEmitter()\n    const hooks = createHooks(emitter)\n    const config = create(Env.DEVELOPMENT, { ...options, port }, userConfigFile)\n    const plugins = await initPlugins(config.plugins, config, hooks)\n\n    logger.debug({\n      label: 'debug',\n      message: `config created ${JSON.stringify(config)}`,\n    })\n\n    if (!noServe) {\n      httpServer = serve(config, hooks)\n\n      staticAssetWatcher = chokidar.watch(config.assets, { ignoreInitial: true }).on('all', () => {\n        hooks.emitBrowserRefresh()\n      })\n    }\n\n    logger.info({\n      label: restarting ? 'restart' : 'start',\n      message: !noServe ? `http://localhost:${config.port}` : '',\n    })\n\n    watchTask = await watch(config, hooks)\n\n    return {\n      config,\n      async close() {\n        emitter.clear()\n        await plugins.cleanup()\n        await staticAssetWatcher.close()\n        await watchTask.close()\n\n        if (httpServer) {\n          await httpServer.close()\n        }\n      },\n    }\n  }\n\n  const configWatcher = chokidar\n    .watch(path.resolve(options.config || defaultConfigFilepath), { ignoreInitial: true })\n    .on('all', async () => {\n      if (restarting) return\n\n      restarting = true\n\n      try {\n        await devServer.close()\n      } catch (e) {\n        console.error(e)\n      }\n\n      console.clear()\n\n      devServer = await startDevServer()\n\n      restarting = false\n    })\n\n  devServer = await startDevServer()\n\n  return {\n    async close() {\n      await configWatcher.close()\n      await devServer.close()\n    },\n  }\n}\n\nexport async function serveCommand(options: PrestaCLIServeOptions) {\n  const configFile = getConfigFile(options.config, true)\n  const port = await getAvailablePort(options.port || configFile.port || 4000)\n\n  const emitter = createEmitter()\n  const hooks = createHooks(emitter)\n  const config = create(Env.PRODUCTION, { ...options, port }, configFile)\n  await initPlugins(config.plugins, config, hooks)\n\n  serve(config, hooks)\n\n  logger.info({\n    label: 'serve',\n    message: `http://localhost:${config.port}`,\n  })\n}\n", "import c, { Kleur } from 'kleur'\n\nimport { Env } from './constants'\n\nexport enum Levels {\n  Debug = 'debug',\n  Info = 'info',\n  Warn = 'warn',\n  Err = 'error',\n}\n\nexport type Message = {\n  level?: Levels\n  label: string | number\n  message?: string\n  duration?: string | number\n  error?: Error\n}\n\nlet logs: any[] = []\n\nconst colors = {\n  [Levels.Debug]: 'magenta',\n  [Levels.Info]: 'blue',\n  [Levels.Warn]: 'yellow',\n  [Levels.Err]: 'red',\n}\n\nexport { c as colors }\n\nexport function getLogs() {\n  if (!process.env.TESTING) {\n    throw new Error('Internal method was called outside test mode')\n  }\n\n  return logs\n}\n\nexport function logger(message: Message) {\n  if (process.env.TESTING) {\n    logs.push(message)\n  } else {\n    const debug = process.env.PRESTA_DEBUG\n    const context = process.env.PRESTA_ENV === Env.PRODUCTION ? 'prod' : 'dev'\n\n    if (!debug && message.level === Levels.Debug) return\n\n    console.log(\n      [\n        c.gray(context),\n        c[colors[message.level || 'info'] as keyof Kleur](message.label),\n        message.message,\n        message.duration ? c.gray('+' + message.duration) : '',\n        message.error ? `\\n\\n${message.error.stack || message.error}\\n\\n` : '',\n      ]\n        .filter(Boolean)\n        .join(' ')\n    )\n  }\n}\n\nexport function debug(message: Message) {\n  logger({ level: Levels.Debug, ...message })\n}\n\nexport function info(message: Message) {\n  logger({ level: Levels.Info, ...message })\n}\n\nexport function warn(message: Message) {\n  logger({ level: Levels.Warn, ...message })\n}\n\nexport function error(message: Message) {\n  logger({ level: Levels.Err, ...message })\n}\n\nexport function raw(...args: any[]) {\n  if (process.env.TESTING) {\n    logs.push(args)\n  } else {\n    console.log(...args)\n  }\n}\n\nexport function newline() {\n  if (process.env.TESTING) return\n  console.log('')\n}\n", "export const Env = {\n  PRODUCTION: 'production',\n  DEVELOPMENT: 'development',\n}\n", "import fs from 'fs'\nimport path from 'path'\nimport getPort from 'get-port'\n\nimport * as logger from './log'\nimport { PrestaCLIDevOptions, PrestaCLIBuildOptions } from './cli'\nimport { Plugin } from './plugins'\n\nexport type Options = {\n  files: string[]\n  output: string\n  assets: string\n  plugins: Plugin[]\n  port: number\n}\n\nexport type Config = Options & {\n  env: string\n  staticOutputDir: string\n  functionsOutputDir: string\n  functionsManifest: string\n}\n\nexport const defaultConfigFilepath = 'presta.config.js'\n\nexport function getAvailablePort(preferred: string) {\n  return getPort({ port: parseInt(preferred, 10) })\n}\n\n/**\n * Fetch a config file. If one was specified by the user, let them know if\n * anything goes wrong. Outside watch mode, this should exit(1) if the user\n * provided a config and there was an error\n */\nexport function getConfigFile(filepath?: string, shouldExit: boolean = false) {\n  const fp = path.resolve(filepath || defaultConfigFilepath)\n\n  try {\n    delete require.cache[fp]\n    return require(fp)\n  } catch (e) {\n    const exists = fs.existsSync(fp)\n\n    // config file exists, should log error, otherwise ignore missing file\n    if (exists) {\n      logger.error({\n        label: 'error',\n        error: e as Error,\n      })\n\n      // we're not in watch mode, exit build\n      if (shouldExit) process.exit(1)\n    }\n\n    return {}\n  }\n}\n\nexport function create(env: string, cli: PrestaCLIBuildOptions | PrestaCLIDevOptions, file: Partial<Options>): Config {\n  const config = {\n    env,\n    output: 'build',\n    assets: 'public',\n    plugins: [],\n    port: 4000,\n    files: [], // TODO where do we validate\n    ...file,\n  }\n\n  // override with CLI\n  if (cli._.length) config.files = cli._\n  if (cli.output) config.output = cli.output\n  if (cli.assets) config.output = cli.assets\n  if (cli.port) config.port = cli.port\n\n  // resolve absolute paths\n  if (config.files) config.files = ([] as string[]).concat(config.files).map((p) => path.resolve(process.cwd(), p))\n  if (config.output) config.output = path.resolve(process.cwd(), config.output)\n  if (config.assets) config.assets = path.resolve(process.cwd(), config.assets)\n\n  return {\n    ...config,\n    staticOutputDir: path.join(config.output, 'static'),\n    functionsOutputDir: path.join(config.output, 'functions'),\n    functionsManifest: path.join(config.output, 'routes.json'),\n  }\n}\n", "import fs from 'fs-extra'\nimport path from 'path'\nimport { create } from 'watch-dependency-graph'\nimport chokidar from 'chokidar'\nimport match from 'picomatch'\nimport merge from 'deep-extend'\n\nimport { outputLambdas } from './outputLambdas'\nimport * as logger from './log'\nimport { getFiles, isStatic, isDynamic } from './getFiles'\nimport { buildStaticFiles, removeBuiltStaticFile, StaticFilesMap } from './renderStaticEntries'\nimport { timer } from './timer'\nimport { Config } from './config'\nimport { Hooks } from './createEmitter'\n\n/*\n * Wraps outputLambdas for logging\n */\nfunction updateLambdas(inputs: string[], config: Config) {\n  const time = timer()\n\n  // always write this, even if inputs = []\n  outputLambdas(inputs, config)\n\n  // if user actually has routes configured, give feedback\n  if (inputs.length) {\n    logger.info({\n      label: 'built',\n      message: `lambdas`,\n      duration: time(),\n    })\n  }\n}\n\nexport function isNewValidFile(file: string, globs: string[], existing: string[]) {\n  return match(globs)(file) && !existing.includes(file)\n}\n\nexport async function watch(config: Config, hooks: Hooks) {\n  let staticFilesMap: StaticFilesMap = {}\n  const files = getFiles(config.files)\n\n  if (!files.length) {\n    logger.warn({\n      label: 'paths',\n      message: 'no files configured',\n    })\n  }\n\n  async function buildFile(file: string, existing: string[], config: Config) {\n    delete require.cache[file]\n\n    // render just file that changed\n    if (isStatic(file)) {\n      const result = await buildStaticFiles([file], config, staticFilesMap)\n      staticFilesMap = merge({}, staticFilesMap, result.staticFilesMap)\n    }\n\n    // update dynamic entry with ALL dynamic files\n    updateLambdas(existing.filter(isDynamic), config)\n  }\n\n  async function buildFiles(files: string[], existing: string[], config: Config) {\n    for (const file of files) {\n      await buildFile(file, existing, config)\n    }\n  }\n\n  /**\n   * Important: if we ever remove initial rendering, we will need to\n   * re-introduce \"file priming\" where we require all files and surface errors\n   * on startup.\n   */\n  await buildFiles(files, files, config)\n  hooks.emitBrowserRefresh()\n\n  /*\n   * Filewatcher watches only presta files. It handles change and remove\n   * events, as well as surfaces dependency tree traversal errors\n   */\n  const fileWatcher = create({ alias: { '@': process.cwd() } })\n\n  fileWatcher.onChange(async (changed) => {\n    await buildFiles(changed, files, config)\n    hooks.emitBrowserRefresh()\n  })\n\n  fileWatcher.onRemove(async ([id]) => {\n    logger.debug({ label: 'watch', message: `removed ${id}` })\n\n    // remove from local hash\n    files.splice(files.indexOf(id), 1)\n\n    // update this regardless, not sure if [id] was dynamic or static\n    updateLambdas(files.filter(isDynamic), config)\n    ;(staticFilesMap[id] || []).forEach((file) => removeBuiltStaticFile(path.join(config.staticOutputDir, file)))\n\n    hooks.emitBrowserRefresh()\n  })\n\n  fileWatcher.onError((e) => {\n    logger.error({\n      label: 'error',\n      error: typeof e === 'string' ? new Error(e) : e,\n    })\n  })\n\n  await fileWatcher.add(files)\n\n  /*\n   * globalWatcher watches the raw file globs passed to the CLI or as `files`\n   * in the config. If checks on add/change to see if a file should be upgraded\n   * to a a Presta source file, and added to the fileWatcher.\n   */\n  const globalWatcher = chokidar.watch(process.cwd(), {\n    ignoreInitial: true,\n    ignored: [config.output, config.assets],\n  })\n\n  globalWatcher.on('add', async (file) => {\n    if (!fs.existsSync(file) || fs.lstatSync(file).isDirectory()) return\n    if (!isNewValidFile(file, config.files, files)) return\n\n    logger.debug({ label: 'watch', message: `add ${file}` })\n\n    files.push(file)\n    await fileWatcher.add(file)\n\n    await buildFile(file, files, config)\n\n    hooks.emitBrowserRefresh()\n  })\n\n  /**\n   * Listens for events from plugins requesting a file to be built\n   */\n  hooks.onBuildFile(async ({ file }) => {\n    await buildFile(file, files, config)\n    hooks.emitBrowserRefresh()\n  })\n\n  return {\n    async close() {\n      await fileWatcher.close()\n      await globalWatcher.close()\n    },\n  }\n}\n", "import fs from 'fs-extra'\nimport path from 'path'\nimport rsort from 'route-sort'\n\nimport { hashContent } from './hashContent'\nimport * as logger from './log'\nimport { Config } from './config'\nimport { Env } from './constants'\n\nfunction slugify(filename: string) {\n  return filename\n    .replace(process.cwd(), '') // /pages/File.page.js\n    .split('.') // [/pages/File, page, js]\n    .reverse()\n    .slice(1)\n    .reverse()\n    .join('-') // /pages/File.page\n    .split('/')\n    .filter(Boolean)\n    .join('-') // pages-File-page\n}\n\nexport function outputLambdas(inputs: string[], config: Config) {\n  const lambdas = inputs\n    .map((input) => {\n      try {\n        const { route } = require(input)\n        const name = slugify(input)\n        const output = path.join(\n          config.functionsOutputDir,\n          config.env === Env.PRODUCTION\n            ? name + '-' + hashContent(fs.readFileSync(input, 'utf8')) + '.js'\n            : name + '.js'\n        )\n\n        logger.debug({\n          label: 'debug',\n          message: `generating ${name} lambda`,\n        })\n\n        // important for watch task\n        delete require.cache[input]\n        delete require.cache[output]\n\n        fs.outputFileSync(\n          output,\n          `import { wrapHandler } from 'presta/dist/wrapHandler';\n      import * as file from '${input}';\n      export const route = file.route\n      export const handler = wrapHandler(file)`\n        )\n\n        return [route, output]\n      } catch (e) {\n        logger.error({\n          label: 'error',\n          error: e as Error,\n        })\n      }\n    })\n    .filter(Boolean) as [string, string][]\n\n  const sorted = rsort(lambdas.map((l) => l[0]))\n  const manifest: { [route: string]: string } = {}\n\n  for (const route of sorted) {\n    const match = lambdas.find((l) => l[0] === route)\n\n    if (match) {\n      manifest[route] = match[1]\n    }\n  }\n\n  fs.outputFileSync(config.functionsManifest, JSON.stringify(manifest))\n\n  return lambdas\n}\n", "export function hashContent(content: string) {\n  var h = 5381,\n    i = content.length\n\n  while (i) h = (h * 33) ^ content.charCodeAt(--i)\n\n  return (h >>> 0).toString(36)\n}\n", "import fs from 'fs-extra'\nimport path from 'path'\nimport globSync from 'tiny-glob/sync'\n\nimport * as logger from './log'\n\nexport function isDynamic(file: string) {\n  return /export\\s.+\\sroute\\s+\\=/.test(fs.readFileSync(file, 'utf-8'))\n}\n\nexport function isStatic(file: string) {\n  return /export\\s.+\\sgetStaticPaths/.test(fs.readFileSync(file, 'utf-8'))\n}\n\nexport function isPrestaFile(file: string) {\n  return isStatic(file) || isDynamic(file)\n}\n\nexport function getFiles(files: string[]): string[] {\n  try {\n    return ([] as string[])\n      .concat(files)\n      .map((file) => globSync(file))\n      .flat()\n      .map((file) => path.resolve(process.cwd(), file)) // make absolute\n  } catch (e) {\n    logger.error({\n      label: 'paths',\n      message: `no files found`,\n      error: e as Error,\n    })\n\n    return []\n  }\n}\n", "import fs from 'fs-extra'\nimport path from 'path'\nimport mime from 'mime-types'\n\nimport * as logger from './log'\nimport { timer } from './timer'\nimport { getRouteParams } from './getRouteParams'\nimport { normalizeResponse } from './normalizeResponse'\nimport { createLiveReloadScript } from './liveReloadScript'\nimport { Env } from './constants'\nimport { Config } from './config'\n\nexport type StaticFilesMap = { [filename: string]: string[] }\n\nexport function pathnameToFile(pathname: string, ext = 'html') {\n  return !!path.extname(pathname)\n    ? pathname // if path has extension, use it\n    : ext === 'html'\n    ? `${pathname}/index.html` // if HTML is inferred, create index\n    : `${pathname}.${ext}` // anything but HTML will need an extension, otherwise browsers will render as text\n}\n\nexport async function removeBuiltStaticFile(file: string) {\n  logger.debug({\n    label: 'debug',\n    message: `removing old static file ${file}`,\n  })\n\n  return fs.remove(file)\n}\n\nexport async function removeBuiltFiles(files: string[]) {\n  return Promise.all(files.map(removeBuiltStaticFile))\n}\n\nexport async function buildStaticFile(file: string, output: string, { footer }: { footer: string }) {\n  const lambda = require(file)\n  const paths = await lambda.getStaticPaths()\n\n  const builtFiles: string[] = []\n\n  if (!paths || !paths.length) return builtFiles\n\n  for (const url of paths) {\n    const time = timer()\n\n    const event = {\n      path: url,\n      pathParameters: lambda.route ? getRouteParams(url, lambda.route) : {},\n    }\n\n    const response = normalizeResponse(await lambda.handler(event, {}))\n    const type = response.headers ? response.headers['Content-Type'] : ''\n    const ext = type ? mime.extension(type as string) || 'html' : 'html'\n    const filename = pathnameToFile(url, ext)\n    const html = response.body + footer\n\n    fs.outputFileSync(path.join(output, filename), html, 'utf-8')\n\n    logger.info({\n      label: 'built',\n      message: url,\n      duration: time(),\n    })\n\n    builtFiles.push(filename)\n  }\n\n  return builtFiles\n}\n\nexport async function buildStaticFiles(files: string[], config: Config, staticFilesMap: StaticFilesMap = {}) {\n  const isDev = config.env === Env.DEVELOPMENT\n  const output = config.staticOutputDir\n  const footer = isDev ? createLiveReloadScript({ port: config.port }) : ''\n\n  for (const file of files) {\n    try {\n      const filename = file.replace(process.cwd(), '')\n      const prevBuiltFiles = staticFilesMap[file] || []\n      const builtFiles = await buildStaticFile(file, output, { footer })\n\n      if (!builtFiles || !builtFiles.length) {\n        logger.warn({\n          label: 'paths',\n          message: `${filename} - no paths to render`,\n        })\n\n        removeBuiltFiles(prevBuiltFiles.map((file) => path.join(output, file)))\n\n        continue\n      }\n\n      // diff and remove files\n      for (const file of prevBuiltFiles) {\n        if (!builtFiles.includes(file)) {\n          removeBuiltStaticFile(path.join(output, file))\n        }\n      }\n\n      staticFilesMap[file] = builtFiles\n    } catch (e) {\n      logger.error({ label: 'error', error: e as Error })\n\n      // exit loop on any error\n      break\n    }\n  }\n\n  return {\n    staticFilesMap,\n  }\n}\n", "export function timer() {\n  const start = process.hrtime()\n  return () => {\n    const [s, nanos] = process.hrtime(start)\n    const ms = nanos / 1000000\n\n    if (s < 1) {\n      return (ms >= 1 ? ms.toFixed(0) : ms.toFixed(2)) + 'ms'\n    } else {\n      return s + '.' + ms.toFixed(0) + 's'\n    }\n  }\n}\n", "import toRegExp from 'regexparam'\n\nimport { PathParameters } from './lambda'\n\n// @see https://github.com/lukeed/regexparam#usage\nexport function getRouteParams(url: string, route: string): PathParameters {\n  const [path] = url.split('?')\n  const result = toRegExp(route)\n  let i = 0\n  let out: PathParameters = {}\n  let matches = result.pattern.exec(path) || []\n\n  while (i < result.keys.length) {\n    out[result.keys[i]] = matches[++i]\n  }\n\n  return out\n}\n", "/**\n * THIS IS PROD CODE, BE CAREFUL WHAT YOU ADD TO THIS FILE\n */\n\nimport { Response as LambdaResponse } from 'lambda-types'\nimport { Response } from './lambda'\n\nfunction stringify(obj: object | string) {\n  return typeof obj === 'object' ? JSON.stringify(obj) : obj\n}\n\nexport function normalizeResponse(response: Partial<Response> | string): LambdaResponse {\n  const {\n    isBase64Encoded = false,\n    statusCode = 200,\n    headers = {},\n    multiValueHeaders = {},\n    body = '',\n    html = undefined,\n    json = undefined,\n    xml = undefined,\n  } = typeof response === 'string'\n    ? {\n        body: response,\n      }\n    : response\n\n  let contentType = 'text/html; charset=utf-8'\n\n  if (!!json) {\n    contentType = 'application/json; charset=utf-8'\n  } else if (!!xml) {\n    contentType = 'application/xml; charset=utf-8'\n  }\n\n  return {\n    isBase64Encoded,\n    statusCode,\n    headers: {\n      'Content-Type': contentType,\n      ...headers,\n    },\n    multiValueHeaders,\n    body: stringify(body || html || json || xml || ''),\n  }\n}\n", "export function createLiveReloadScript({ port }: { port: number }) {\n  return `\n    <script>\n      (function (global) {\n        var socket = new WebSocket('ws://localhost:${port}');\n\n        socket.addEventListener('open', function (event) {\n          console.log('[presta] connected on port ${port}')\n        });\n\n        socket.addEventListener('message', function (event) {\n          console.log(\\`'[presta] received \\$\\{event.data\\}\\`)\n          if (event.data === 'refresh') {\n            global.location.reload();\n          }\n        });\n\n        socket.addEventListener('close', function () {\n          console.log('[presta] disconnected')\n        });\n      })(this);\n    </script>\n  `\n}\n", "import * as logger from './log'\nimport { Hooks } from './createEmitter'\nimport { Config } from './config'\n\nexport type PluginInterface = {\n  cleanup?(): void\n}\nexport type Plugin = (config: Config, hooks: Hooks) => Promise<PluginInterface | void> | PluginInterface | void\nexport type PluginInit = (...props: any) => Plugin\n\nexport function createPlugin(init: PluginInit) {\n  return init\n}\n\nexport async function initPlugins(plugins: Plugin[], instance: Config, hooks: Hooks) {\n  const instantiated = await Promise.all(\n    plugins\n      .map((p) => {\n        try {\n          return p(instance, hooks)\n        } catch (e) {\n          logger.error({\n            label: 'error',\n            error: e as Error,\n          })\n        }\n      })\n      .filter(Boolean) as PluginInterface[]\n  )\n\n  return {\n    async cleanup() {\n      return Promise.all(instantiated.map((p) => p && p.cleanup && p.cleanup()))\n    },\n  }\n}\n", "export enum Events {\n  PostBuild = 'post-build',\n  BuildFile = 'build-file',\n  BrowserRefresh = 'browser-refresh',\n}\n\nexport type Callable = (...args: any[]) => void\n\nexport type HookPostBuildPayload = {\n  output: string\n  staticOutput: string\n  functionsOutput: string\n  functionsManifest: Record<string, string>\n}\n\nexport type HookBuildFilePayload = {\n  file: string\n}\n\nexport type DestroyHookCallback = () => void\n\nexport type Hooks = {\n  emitPostBuild(props: HookPostBuildPayload): void\n  onPostBuild(cb: (props: HookPostBuildPayload) => void): DestroyHookCallback\n  emitBuildFile(props: HookBuildFilePayload): void\n  onBuildFile(cb: (props: HookBuildFilePayload) => void): DestroyHookCallback\n  emitBrowserRefresh(): void\n  onBrowserRefresh(cb: () => void): DestroyHookCallback\n}\n\nexport function createEmitter() {\n  let events: { [event: string]: Callable[] } = {}\n\n  function emit(ev: string, ...args: any[]): void {\n    events[ev] ? events[ev].map((fn: Callable) => fn(...args)) : []\n  }\n\n  function on(ev: string, fn: (...args: any[]) => void) {\n    events[ev] = events[ev] ? events[ev].concat(fn) : [fn]\n    return () => events[ev].splice(events[ev].indexOf(fn), 1)\n  }\n\n  function clear() {\n    events = {}\n  }\n\n  function listeners(ev: string) {\n    return events[ev] || []\n  }\n\n  return {\n    emit,\n    on,\n    clear,\n    listeners,\n  }\n}\n\nexport function createHooks(emitter: ReturnType<typeof createEmitter>): Hooks {\n  return {\n    emitPostBuild(props) {\n      emitter.emit('postBuild', props)\n    },\n    onPostBuild(cb) {\n      return emitter.on('postBuild', cb)\n    },\n    emitBuildFile(props) {\n      emitter.emit('buildFile', props)\n    },\n    onBuildFile(cb) {\n      return emitter.on('buildFile', cb)\n    },\n    emitBrowserRefresh() {\n      emitter.emit('browserRefresh')\n    },\n    onBrowserRefresh(cb) {\n      return emitter.on('browserRefresh', cb)\n    },\n  }\n}\n", "import path from 'path'\nimport fs from 'fs-extra'\nimport { build as esbuild } from 'esbuild'\n\nimport { outputLambdas } from './outputLambdas'\nimport { getFiles, isStatic, isDynamic } from './getFiles'\nimport { buildStaticFiles } from './renderStaticEntries'\nimport { timer } from './timer'\nimport * as logger from './log'\nimport { Config } from './config'\nimport { Hooks } from './createEmitter'\nimport { requireSafe } from './utils'\n\nexport async function build(config: Config, hooks: Hooks) {\n  const totalTime = timer()\n  const files = getFiles(config.files)\n  const staticIds = files.filter(isStatic)\n  const dynamicIds = files.filter(isDynamic)\n\n  logger.debug({\n    label: 'build',\n    message: 'starting build',\n  })\n\n  if (!staticIds.length && !dynamicIds.length) {\n    logger.warn({\n      label: 'files',\n      message: 'no files were found, nothing to build',\n    })\n  } else {\n    let staticTime = ''\n    let staticFileAmount = 0\n    let dynamicTime = ''\n    let copyTime = ''\n\n    const tasks = await Promise.allSettled([\n      (async () => {\n        if (staticIds.length) {\n          const time = timer()\n\n          const { staticFilesMap } = await buildStaticFiles(staticIds, config)\n\n          staticTime = time()\n          staticFileAmount = Object.keys(staticFilesMap).reduce((count, key) => {\n            return (count += staticFilesMap[key].length)\n          }, 0)\n        }\n      })(),\n      (async () => {\n        if (dynamicIds.length) {\n          const time = timer()\n          const pkg = requireSafe(path.join(process.cwd(), 'package.json'))\n\n          outputLambdas(dynamicIds, config)\n\n          await esbuild({\n            entryPoints: Object.values(require(config.functionsManifest)),\n            outdir: config.functionsOutputDir,\n            platform: 'node',\n            target: ['node12'],\n            minify: true,\n            allowOverwrite: true,\n            external: Object.keys(pkg.dependencies || {}),\n            bundle: true,\n            define: {\n              'process.env.PRESTA_SERVERLESS_RUNTIME': 'true',\n            },\n          })\n\n          dynamicTime = time()\n        }\n      })(),\n      (async () => {\n        if (fs.existsSync(config.assets)) {\n          const time = timer()\n\n          fs.copySync(config.assets, config.staticOutputDir)\n\n          copyTime = time()\n        }\n      })(),\n    ])\n\n    // since we're building (not watch) if any task fails, exit with error\n    if (tasks.find((task) => task.status === 'rejected')) {\n      logger.debug({\n        label: 'build',\n        message: 'build partially failed',\n      })\n\n      // log out errors\n      tasks.forEach((task) => {\n        if (task.status === 'rejected') {\n          // TODO can swallow errors in testing\n          logger.error({\n            label: 'error',\n            error: task.reason,\n          })\n        }\n      })\n\n      throw new Error('presta build failed')\n    }\n\n    if (staticTime) {\n      logger.info({\n        label: 'static',\n        message: `rendered ${staticFileAmount} file(s)`,\n        duration: staticTime,\n      })\n    }\n\n    if (dynamicTime) {\n      logger.info({\n        label: 'lambda',\n        message: `compiled ${dynamicIds.length} function(s)`,\n        duration: dynamicTime,\n      })\n    }\n\n    if (copyTime) {\n      logger.info({\n        label: 'assets',\n        message: `copied in ${copyTime}`,\n      })\n    }\n\n    hooks.emitPostBuild({\n      output: config.output,\n      staticOutput: config.staticOutputDir,\n      functionsOutput: config.functionsOutputDir,\n      functionsManifest: requireSafe(config.functionsManifest),\n    })\n\n    if (staticTime || dynamicTime) {\n      logger.info({\n        label: 'complete',\n        message: `in ${totalTime()}`,\n      })\n    }\n  }\n}\n", "export function requireFresh(mod: string) {\n  delete require.cache[mod]\n  return require(mod)\n}\n\nexport function requireSafe(mod: string) {\n  try {\n    return requireFresh(mod)\n  } catch (e) {\n    return {}\n  }\n}\n", "import { Socket } from 'net'\nimport http from 'http'\nimport sirv from 'sirv'\nimport mime from 'mime-types'\nimport toRegExp from 'regexparam'\nimport status from 'statuses'\nimport { WebSocketServer } from 'ws'\n\nimport { timer } from './timer'\nimport * as logger from './log'\nimport { createDefaultHtmlResponse } from './createDefaultHtmlResponse'\nimport { requestToEvent } from './requestToEvent'\nimport { sendServerlessResponse } from './sendServerlessResponse'\nimport { createLiveReloadScript } from './liveReloadScript'\nimport { Handler, Event, Response, Context } from './lambda'\nimport { Config } from './config'\nimport { Hooks } from './createEmitter'\nimport { normalizeResponse } from './normalizeResponse'\nimport { requireFresh } from './utils'\n\nexport interface HttpError extends Error {\n  statusCode?: number\n  message: string\n}\n\nexport function createHttpError(statusCode: number, message: string): HttpError {\n  const error = new Error(message)\n  // @ts-ignore\n  error.statusCode = statusCode\n  return error\n}\n\nexport function getMimeType(response: Response) {\n  const type = (response?.headers || {})['Content-Type'] || 'html'\n  return type ? mime.extension(String(type)) || 'html' : 'html'\n}\n\nexport function loadLambdaFroManifest(url: string, manifest: { [route: string]: string }): { handler: Handler } {\n  const routes = Object.keys(manifest)\n  const lambdaFilepath = routes\n    .map((route) => ({\n      matcher: toRegExp(route),\n      route,\n    }))\n    .filter(({ matcher }) => {\n      return matcher.pattern.test(url.split('?')[0])\n    })\n    .map(({ route }) => manifest[route])[0]\n\n  return lambdaFilepath ? require(lambdaFilepath) : undefined\n}\n\nexport async function processHandler(event: Event, lambda: { handler: Handler }) {\n  const accept = event.headers.Accept || event.headers.accept\n  const acceptsJson = accept && accept.includes('json')\n\n  /*\n   * No asset file, no static file, try dynamic\n   */\n  try {\n    if (!lambda || !lambda.handler) {\n      throw createHttpError(404, '')\n    }\n\n    return normalizeResponse(await lambda.handler(event, { awsRequestId: 'presta dev' } as Context))\n  } catch (e) {\n    const error = e as HttpError\n    const { statusCode = 500 } = error\n\n    if (statusCode > 499)\n      logger.error({\n        label: 'error',\n        message: error.message || status.message[statusCode],\n        error,\n      })\n\n    return normalizeResponse({\n      statusCode: statusCode,\n      html: acceptsJson ? undefined : createDefaultHtmlResponse({ statusCode }),\n      json: acceptsJson ? { detail: status.message[statusCode] } : undefined,\n    })\n  }\n}\n\nexport function createRequestHandler({ port, config }: { port: number; config: Config }) {\n  return async function requestHandler(req: http.IncomingMessage, res: http.ServerResponse) {\n    const time = timer()\n    const event = await requestToEvent(req) // stock AWS Event shape\n    const manifest = requireFresh(config.functionsManifest)\n    const lambda = loadLambdaFroManifest(event.path, manifest)\n    const response = await processHandler(event, lambda)\n    const redir = response.statusCode > 299 && response.statusCode < 399\n    const mime = getMimeType(response)\n\n    if (mime === 'html') {\n      response.body = (response.body || '').split('</body>')[0] + createLiveReloadScript({ port })\n    }\n\n    logger[response.statusCode < 299 ? 'info' : 'error']({\n      label: 'serve',\n      message: `${response.statusCode} ${redir ? response?.headers?.Location || event.path : event.path}`,\n      duration: time(),\n    })\n\n    sendServerlessResponse(res, response)\n  }\n}\n\nexport function createServerHandler({ port, config }: { port: number; config: Config }) {\n  const staticDir = config.staticOutputDir\n  const assetDir = config.assets\n\n  return async function serveHandler(req: http.IncomingMessage, res: http.ServerResponse) {\n    const time = timer()\n    const url = req.url as string\n\n    logger.debug({\n      label: 'debug',\n      message: `handling ${url}`,\n    })\n\n    // hook into sirv for logging only\n    function setHeaders(res: http.ServerResponse, pathname: string) {\n      logger.info({\n        label: 'serve',\n        message: `${res.statusCode} ${pathname}`,\n        duration: time(),\n      })\n    }\n\n    sirv(assetDir, { dev: true, setHeaders })(req, res, () => {\n      sirv(staticDir, { dev: true, setHeaders })(req, res, async () => {\n        createRequestHandler({ port, config })(req, res)\n      })\n    })\n  }\n}\n\nexport function serve(config: Config, hooks: Hooks) {\n  const port = config.port\n  const server = http.createServer(createServerHandler({ port, config })).listen(port)\n  const websocket = new WebSocketServer({ server })\n  const sockets: Socket[] = []\n\n  server.on('connection', (socket) => {\n    sockets.push(socket)\n    socket.on('close', () => sockets.splice(sockets.indexOf(socket), 1))\n  })\n\n  hooks.onBrowserRefresh(() => {\n    logger.debug({\n      label: 'debug',\n      message: `refresh event received`,\n    })\n\n    websocket.clients.forEach((ws) => ws.send('refresh'))\n  })\n\n  return {\n    async close() {\n      // so just always resolve OK\n      await new Promise((y) => {\n        server.close(() => y(1))\n        // sockets includes ws sockets\n        sockets.forEach((ws) => ws.destroy())\n      })\n    },\n  }\n}\n", "import status from 'statuses'\n\nexport function createDefaultHtmlResponse({ statusCode }: { statusCode: number }) {\n  return `<!-- built with presta https://npm.im/presta -->\n    <!DOCTYPE html>\n    <html>\n      <head>\n        <meta charset=\"UTF-8\" />\n        <meta name=\"viewport\" content=\"width=device-width,initial-scale=1\" />\n        <title>${statusCode} \u2014\u00A0${status.message[statusCode]}</title>\n        <link rel=\"icon\" type=\"image/png\" href=\"https://presta.run/favicon.png\">\n        <link rel=\"icon\" type=\"image/svg\" href=\"https://presta.run/favicon.svg\">\n        <link rel=\"preconnect\" href=\"https://fonts.googleapis.com\">\n        <link rel=\"preconnect\" href=\"https://fonts.gstatic.com\" crossorigin>\n        <link href=\"https://fonts.googleapis.com/css2?family=Inter:wght@400;900&display=swap\" rel=\"stylesheet\"> \n        <link rel='stylesheet' href='https://unpkg.com/svbstrate@5.1.0/svbstrate.css' />\n        <style>\n          html,body {\n            font-family: 'Inter', 'sans-serif';\n            color: #23283D;\n            background-color: #DADEF0;\n          }\n          #favicon {\n            fill: #23283D;\n          }\n          @media (prefers-color-scheme: dark) {\n            html,body {\n              color: #DADEF0;\n              background-color: #23283D;\n            }\n            #favicon {\n              fill: #DADEF0;\n            }\n          }\n        </style>\n      </head>\n      <body class='w f aic jcc' style='height: 100vh'>\n        <div class='p12 tac'>\n          <h1>${statusCode}</h1>\n          <p class='mb1'>${status.message[statusCode]}</p>\n\n          <svg width=\"32\" height=\"32\" viewBox=\"0 0 32 32\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n            <g clip-path=\"url(#a)\">\n              <path id=\"favicon\" fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M10.4 7c-.3 0-.8.2-1 .5L1.1 22.1c-.2.3 0 .6.3.6l4 .3-2.1 2.6c-.2.3-.1.6.2.6l16.8 1.3c.4 0 .8-.2 1-.4L32 13.9c.2-.2.1-.5-.2-.5l-6.4-.5 2.2-4c.2-.3 0-.5-.3-.6L10.4 7ZM24 12.8l1.9-3.4-15.5-1.2-7.7 13.4 3.6.3 7.5-9.4c.3-.3.7-.5 1-.4l9.2.7ZM7.6 22l7.1-8.9 8.7.7-5.2 9L7.6 22Zm-1 1.1 11.6 1c.3 0 .8-.3 1-.6l5.5-9.6 5.5.5-9.7 12L5 25.2l1.7-2Z\" fill=\"#23283D\"/>\n            </g>\n            <defs>\n              <clipPath id=\"a\">\n                <path fill=\"#fff\" d=\"M0 0h32v32H0z\"/>\n              </clipPath>\n            </defs>\n          </svg>\n        </div>\n      </body>\n    </html>\n  `\n}\n", "import http from 'http'\nimport { parse as parseUrl } from 'url'\nimport rawBody from 'raw-body'\nimport mime from 'mime-types'\n\nimport { Event } from './lambda'\nimport { normalizeHeaders } from './normalizeHeaders'\nimport { parseQueryStringParameters } from './parseQueryStringParameters'\n\n// @see https://github.com/netlify/cli/blob/27bb7b9b30d465abe86f87f4274dd7a71b1b003b/src/utils/serve-functions.js#L167\nconst BASE_64_MIME_REGEXP = /image|audio|video|application\\/pdf|application\\/zip|applicaton\\/octet-stream/i\nfunction shouldBase64Encode(contentType: string) {\n  return Boolean(contentType) && BASE_64_MIME_REGEXP.test(contentType)\n}\n\nexport async function requestToEvent(req: http.IncomingMessage): Promise<Event> {\n  const { url: path = '', method } = req\n  const { headers, multiValueHeaders } = normalizeHeaders(req.headers)\n  const isBase64Encoded = shouldBase64Encode(headers['content-type'] || '')\n  const contentLengthHeader = headers['content-length']\n  const body = contentLengthHeader\n    ? await rawBody(req, {\n        limit: '1mb',\n        encoding: headers['content-type'] ? mime.charset(headers['content-type']) || true : true,\n      })\n    : undefined\n  const rawQuery = parseUrl(path).query || ''\n  const { queryStringParameters, multiValueQueryStringParameters } = parseQueryStringParameters(rawQuery)\n\n  return {\n    rawUrl: path,\n    path,\n    httpMethod: method as string,\n    headers,\n    multiValueHeaders,\n    rawQuery,\n    queryStringParameters,\n    multiValueQueryStringParameters,\n    body: body ? Buffer.from(body).toString(isBase64Encoded ? 'base64' : 'utf8') : null,\n    isBase64Encoded,\n    pathParameters: {},\n  }\n}\n", "import http from 'http'\nimport { Headers, MultiValueHeaders } from './lambda'\n\nexport function normalizeHeaders(rawHeaders: http.IncomingMessage['headers']) {\n  const headers: Headers = {}\n  const multiValueHeaders: MultiValueHeaders = {}\n\n  for (const header of Object.keys(rawHeaders)) {\n    const key = header.toLowerCase()\n    const value = rawHeaders[header]\n\n    if (!value) continue\n\n    if (Array.isArray(value)) {\n      multiValueHeaders[key] = value\n    } else {\n      headers[key] = value\n    }\n  }\n\n  return { headers, multiValueHeaders }\n}\n", "/**\n * THIS IS PROD CODE, BE CAREFUL WHAT YOU ADD TO THIS FILE\n */\n\nimport { parse as parseQuery } from 'query-string'\nimport { QueryStringParameters, MultiValueQueryStringParameters } from './lambda'\n\nexport function parseQueryStringParameters(query: string) {\n  const params = parseQuery(query, { arrayFormat: 'comma' })\n\n  const queryStringParameters: QueryStringParameters = {}\n  const multiValueQueryStringParameters: MultiValueQueryStringParameters = {}\n\n  for (const param of Object.keys(params)) {\n    const value = params[param]\n    if (Array.isArray(value)) {\n      multiValueQueryStringParameters[param] = value\n    } else if (value) {\n      queryStringParameters[param] = value\n    }\n  }\n\n  return { queryStringParameters, multiValueQueryStringParameters }\n}\n", "import http from 'http'\n\nimport { normalizeResponse } from './normalizeResponse'\nimport { Response } from './lambda'\n\nexport function sendServerlessResponse(res: http.ServerResponse, r: Partial<Response>) {\n  const response = normalizeResponse(r)\n\n  // @see https://github.com/netlify/cli/blob/27bb7b9b30d465abe86f87f4274dd7a71b1b003b/src/utils/serve-functions.js#L73\n  for (const key in r.multiValueHeaders) {\n    res.setHeader(key, String(r.multiValueHeaders[key]))\n  }\n\n  for (const key in r.headers) {\n    res.setHeader(key, String(r.headers[key]))\n  }\n\n  res.statusCode = response.statusCode\n  res.write(response.body)\n  res.end()\n}\n"],
  "mappings": "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",
  "names": []
}
