UNPKG

99.2 kBJavaScriptView Raw
1#!/usr/bin/env node
2var Ye=Object.create;var L=Object.defineProperty,et=Object.defineProperties,tt=Object.getOwnPropertyDescriptor,rt=Object.getOwnPropertyDescriptors,ot=Object.getOwnPropertyNames,de=Object.getOwnPropertySymbols,st=Object.getPrototypeOf,fe=Object.prototype.hasOwnProperty,nt=Object.prototype.propertyIsEnumerable;var ge=(e,t,r)=>t in e?L(e,t,{enumerable:!0,configurable:!0,writable:!0,value:r}):e[t]=r,f=(e,t)=>{for(var r in t||(t={}))fe.call(t,r)&&ge(e,r,t[r]);if(de)for(var r of de(t))nt.call(t,r)&&ge(e,r,t[r]);return e},R=(e,t)=>et(e,rt(t)),it=e=>L(e,"__esModule",{value:!0});var at=(e,t,r)=>{if(t&&typeof t=="object"||typeof t=="function")for(let o of ot(t))!fe.call(e,o)&&o!=="default"&&L(e,o,{get:()=>t[o],enumerable:!(r=tt(t,o))||r.enumerable});return e},c=e=>at(it(L(e!=null?Ye(st(e)):{},"default",e&&e.__esModule&&"default"in e?{get:()=>e.default,enumerable:!0}:{value:e,enumerable:!0})),e);var Je=c(require("fs-extra")),Ze=c(require("sade"));var lt="presta",ct="0.41.1",pt="Hyper minimal framework for the modern web.",ut="index.js",mt="index.d.ts",dt={presta:"cli.js"},ft={build:"node scripts/build",cloc:"pnpx cloc lib/*.ts",typecheck:"pnpx tsc --noEmit"},gt={type:"git",url:"git+ssh://git@github.com/sure-thing/presta.git"},ht="estrattonbailey",bt="MIT",yt={url:"https://github.com/sure-thing/presta/issues"},vt="https://github.com/sure-thing/presta#readme",xt={"@netlify/functions":"^0.7.2","@types/mime-types":"^2.1.0",chokidar:"^3.4.3",deepmerge:"^4.2.2",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","pocket.io":"^0.1.4","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.0"},wt={"@types/fs-extra":"^9.0.12","@types/picomatch":"^2.2.4","@types/sade":"^1.7.3","@types/statuses":"^2.0.0",proxyquire:"^2.1.3","supertest-fetch":"^1.4.3"},he={name:lt,version:ct,description:pt,main:ut,types:mt,bin:dt,scripts:ft,repository:gt,author:ht,license:bt,bugs:yt,homepage:vt,dependencies:xt,devDependencies:wt};var D=c(require("kleur"));var b={PRODUCTION:"production",DEVELOPMENT:"development"};var St={pid:process.pid,cwd:process.cwd(),env:b.PRODUCTION,debug:!1};function U(e){return global.__presta__=e,e}function E(){return global.__presta__||U(St),global.__presta__}var w;(function(n){n.Debug="debug",n.Info="info",n.Warn="warn",n.Err="error"})(w||(w={}));var be=[],Pt={[w.Debug]:"magenta",[w.Info]:"blue",[w.Warn]:"yellow",[w.Err]:"red"};function V(e){if(process.env.TESTING)be.push(e);else{let t=E().debug,r=E().env===b.PRODUCTION?"prod":"dev";if(!t&&e.level===w.Debug)return;console.log([D.default.gray(r),D.default[Pt[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 u(e){V(f({level:w.Debug},e))}function y(e){V(f({level:w.Info},e))}function O(e){V(f({level:w.Warn},e))}function d(e){V(f({level:w.Err},e))}function z(...e){process.env.TESTING?be.push(e):console.log(...e)}function F(){process.env.TESTING||console.log("")}var ve=c(require("fs")),S=c(require("path"));function ye(){let e={};function t(i,...l){e[i]&&e[i].map(s=>s(...l))}function r(i,l){return e[i]=e[i]?e[i].concat(l):[l],()=>e[i].splice(e[i].indexOf(l),1)}function o(){e={}}function n(i){return e[i]||[]}return{emit:t,on:r,clear:o,listeners:n}}var xe="presta.config.js";function we(e,{cwd:t}){return e.files&&(e.files=[].concat(e.files).map(r=>S.default.resolve(t,r))),e.output&&(e.output=S.default.resolve(t,e.output)),e.assets&&(e.assets=S.default.resolve(t,e.assets)),e}function j(e,t=!1){let r=S.default.resolve(e||xe);try{return require(r)}catch(o){return ve.default.existsSync(r)&&(d({label:"error",error:o}),t&&process.exit(1)),{}}}async function Se(){return u({label:"debug",message:"config file values cleared"}),U(await $(R(f({},E()),{config:{}})))}async function $({cwd:e=process.cwd(),env:t=E().env,config:r={},cli:o={}}){r=we(f({},r),{cwd:e}),o=we(f({},o),{cwd:e});let n={output:S.default.resolve(e,o.output||r.output||"build"),assets:S.default.resolve(o.assets||r.assets||"public"),files:o.files&&o.files.length?o.files:r.files?[].concat(r.files):[]},i=o.port?parseInt(o.port):r.port||4e3,l=E(),s=l.events||ye();s.clear();let a=U(R(f(f({},l),n),{env:t,cwd:e,port:i,debug:o.debug||E().debug,configFilepath:S.default.resolve(o.config||xe),staticOutputDir:S.default.join(n.output,"static"),functionsOutputDir:S.default.join(n.output,"functions"),functionsManifest:S.default.join(n.output,"routes.json"),events:s,hooks:{emitPostBuild(p){s.emit("postBuild",p)},onPostBuild(p){return s.on("postBuild",p)},emitBuildFile(p){s.emit("buildFile",p)},onBuildFile(p){return s.on("buildFile",p)},emitBrowserRefresh(){s.emit("browserRefresh")},onBrowserRefresh(p){return s.on("browserRefresh",p)}}}));return r.plugins&&await Promise.all(r.plugins.map(p=>{try{return p(E)}catch(m){d({label:"error",error:m})}})),u({label:"debug",message:`config created ${JSON.stringify(a)}`}),a}var Z=c(require("fs-extra")),je=c(require("watch-dependency-graph")),Ie=c(require("chokidar")),He=c(require("picomatch"));var G=c(require("fs-extra")),oe=c(require("path")),Ee=c(require("route-sort"));function Pe(e){for(var t=5381,r=e.length;r;)t=t*33^e.charCodeAt(--r);return(t>>>0).toString(36)}function Et(e,t){let{route:r}=require(e),o=oe.default.basename(e).split(".").reverse().slice(1).reverse().join("."),n=oe.default.join(t.functionsOutputDir,t.env===b.PRODUCTION?o+"-"+Pe(G.default.readFileSync(e,"utf8"))+".js":o+".js");return u({label:"debug",message:`generating ${o} lambda`}),delete require.cache[e],delete require.cache[n],G.default.outputFileSync(n,`import { wrapHandler } from 'presta';
7import * as file from '${e}';
8export const route = file.route
9export const handler = wrapHandler(file)`),[r,n]}function Q(e,t){let r=e.map(i=>{try{return Et(i,t)}catch(l){return d({label:"error",error:l}),null}}).filter(Boolean),o=(0,Ee.default)(r.map(i=>i[0])),n={};for(let i of o){let l=r.find(s=>s[0]===i);l&&(n[i]=l[1])}return G.default.outputFileSync(t.functionsManifest,JSON.stringify(n)),r}var se=c(require("fs-extra")),Fe=c(require("path")),Ce=c(require("tiny-glob/sync"));function T(e){return/export\s.+\sroute\s+\=/.test(se.default.readFileSync(e,"utf-8"))}function M(e){return/export\s.+\sgetStaticPaths/.test(se.default.readFileSync(e,"utf-8"))}function _(e){try{return[].concat(e.files).map(t=>(0,Ce.default)(t,{cwd:e.cwd})).flat().map(t=>Fe.default.resolve(e.cwd,t))}catch(t){return d({label:"paths",message:"no files found",error:t}),[]}}var $e=c(require("fs-extra")),ne=c(require("path")),Te=c(require("mime-types"));function P(){let e=process.hrtime();return()=>{let[t,r]=process.hrtime(e),o=r/1e6;return t<1?(o>=1?o.toFixed(0):o.toFixed(2))+"ms":t+"."+o.toFixed(0)+"s"}}var ke=c(require("regexparam"));function Re(e,t){let[r]=e.split("?"),o=(0,ke.default)(t),n=0,i={},l=o.pattern.exec(r)||[];for(;n<o.keys.length;)i[o.keys[n]]=l[++n];return i}function Ft(e){return typeof e=="object"?JSON.stringify(e):e}function C(e){let{isBase64Encoded:t=!1,statusCode:r=200,headers:o={},multiValueHeaders:n={},body:i="",html:l=void 0,json:s=void 0,xml:a=void 0}=typeof e=="string"?{body:e}:e,p="text/html; charset=utf-8";return s?p="application/json; charset=utf-8":a&&(p="application/xml; charset=utf-8"),{isBase64Encoded:t,statusCode:r,headers:f({"Content-Type":p},o),multiValueHeaders:n,body:Ft(i||l||s||a||"")}}var A={};var De=c(require("fs-extra")),Oe=c(require("path"));function W(e,t){u({label:"debug",message:`removing old static file ${e}`}),De.default.remove(Oe.default.join(t.staticOutputDir,e))}function J({port:e}){return`
10 <script>
11 (function (global) {
12 try {
13 const socketio = document.createElement('script')
14 socketio.src = 'https://unpkg.com/pocket.io@0.1.4/min.js'
15 socketio.onload = function init () {
16 var disconnected = false
17 var socket = io('http://localhost:${e}', {
18 reconnectionAttempts: 3
19 })
20 socket.on('connect', function() { console.log('presta connected on port ${e}') })
21 socket.on('refresh', function() {
22 global.location.reload()
23 })
24 socket.on('disconnect', function() {
25 disconnected = true
26 })
27 socket.on('reconnect_failed', function(e) {
28 if (disconnected) return
29 console.error("presta - connection to server on :${e} failed")
30 })
31 }
32 document.head.appendChild(socketio)
33 } catch (e) {}
34 })(this);
35 <\/script>
36 `}function Ct(e,t="html"){return ne.default.extname(e)?e:t==="html"?`${e}/index.html`:`${e}.${t}`}function q(e,t){return new Promise(async(r,o)=>{u({label:"debug",message:`rendering ${JSON.stringify(e)}`});let n=[],i=J({port:t.port});for(let l of e){let s=l.replace(t.cwd,"");try{delete require.cache[l];let a=require(l),p=await a.getStaticPaths(),m=A[l]=A[l]||[],k=[];if(!p||!p.length){O({label:"paths",message:`${s} - no paths to render`}),m.forEach(g=>W(g,t));continue}for(let g of p){let I=P(),ue={path:g,routeParameters:a.route?Re(g,a.route):{}},H=C(await a.handler(ue,{})),v=H.headers?H.headers["Content-Type"]:"",x=v&&Te.default.extension(v)||"html",B=Ct(g,x),re=H.body+(t.env===b.PRODUCTION?"":i);$e.default.outputFileSync(ne.default.join(t.staticOutputDir,B),re,"utf-8"),n.push(B),k.push(B),y({label:"built",message:g,duration:I()})}for(let g of m)k.includes(g)||W(g,t);A[l]=k}catch(a){t.env==="development"?(d({label:"error",message:"errors detected, pausing...",error:a}),r({allGeneratedFiles:n})):(d({label:"error",error:a}),o(a));break}}r({allGeneratedFiles:n})})}function K(e,t){let r=P();Q(e,t),e.length&&y({label:"built",message:"lambdas",duration:r()})}async function Be(e){let t=_(e),r=Z.default.existsSync(e.configFilepath);t.length||O({label:"paths",message:"no files configured"}),K(t.filter(T),e);let o=(0,je.create)({alias:{"@":e.cwd}}),n=Ie.default.watch(e.cwd,{ignoreInitial:!0,ignored:[e.output,e.assets]});async function i(){t=_(e),await q(t.filter(M),e),K(t.filter(T),e)}async function l(s){M(s)&&await q([s],e),T(s)&&K(t.filter(T),e),e.hooks.emitBrowserRefresh()}e.hooks.onBuildFile(({file:s})=>{l(s)}),o.onRemove(async([s])=>{u({label:"watch",message:`fileWatcher - removed ${s}`}),t.splice(t.indexOf(s),1),K(t.filter(T),e),s===e.configFilepath&&(e=await Se(),r=!1,i()),(A[s]||[]).forEach(a=>W(a,e))}),o.onChange(async s=>{for(let a of s)if(u({label:"watch",message:`fileWatcher - changed ${a}`}),a===e.configFilepath){delete require.cache[e.configFilepath];try{e=await $({config:j(e.configFilepath)}),i()}catch(p){d({label:"error",error:p})}}else l(a)}),o.onError(s=>{d({label:"error",error:typeof s=="string"?new Error(s):s})}),n.on("all",async(s,a)=>{if(!(!/add|change/.test(s)||!Z.default.existsSync(a)||Z.default.lstatSync(a).isDirectory())&&((0,He.default)(e.files)(a)&&!t.includes(a)&&(u({label:"watch",message:`globalWatcher - add ${a}`}),t.push(a),await o.add(a),l(a)),a===e.configFilepath&&!r)){u({label:"watch",message:`globalWatcher - add config file ${a}`}),await o.add(e.configFilepath);try{e=await $({config:j(e.configFilepath)}),r=!0,i()}catch(p){d({label:"error",error:p})}}}),await o.add(t),r&&await o.add(e.configFilepath);try{t.map(require)}catch(s){d({label:"error",error:s})}}var ie=c(require("fs-extra")),Ae=c(require("esbuild"));function kt(e){try{return require(e)}catch(t){return{}}}async function Me(e){let t=P(),r=_(e),o=r.filter(M),n=r.filter(T);if(u({label:"build",message:"starting build"}),!o.length&&!n.length)O({label:"files",message:"no files were found, nothing to build"});else{let i="",l=0,s="",a="",p=await Promise.allSettled([(async()=>{if(o.length){let m=P(),{allGeneratedFiles:k}=await q(o,e);i=m(),l=k.length}})(),(async()=>{if(n.length){let m=P();Q(n,e),await(0,Ae.build)({entryPoints:Object.values(require(e.functionsManifest)),outdir:e.functionsOutputDir,bundle:!0,platform:"node",target:["node12"],minify:!0,allowOverwrite:!0,define:{"process.env.PRESTA_SERVERLESS_RUNTIME":"true"}}),s=m()}})(),(async()=>{if(ie.default.existsSync(e.assets)){let m=P();ie.default.copySync(e.assets,e.staticOutputDir),a=m()}})()]);if(p.find(m=>m.status==="rejected")){u({label:"build",message:"build partially failed"}),p.forEach(m=>{m.status==="rejected"&&d({label:"error",error:m.reason})}),process.exit(1);return}(i||s)&&F(),i&&y({label:"static",message:`rendered ${l} file(s)`,duration:i}),s&&y({label:"lambda",message:`compiled ${n.length} function(s)`,duration:s}),a&&y({label:"assets",message:`copied in ${a}`}),e.hooks.emitPostBuild({output:e.output,staticOutput:e.staticOutputDir,functionsOutput:e.functionsOutputDir,functionsManifest:kt(e.functionsManifest)}),(i||s)&&(F(),y({label:"complete",message:`in ${t()}`}),F())}}var Tt=c(require("fs")),jt=c(require("path")),Ue=c(require("http")),Ve=c(require("get-port")),le=c(require("sirv")),ze=c(require("chokidar")),Ge=c(require("mime-types")),Qe=c(require("regexparam")),ee=c(require("statuses"));var ae=c(require("statuses"));function X({statusCode:e}){return`<!-- built with presta https://npm.im/presta -->
37 <!DOCTYPE html>
38 <html>
39 <head>
40 <meta charset="UTF-8" />
41 <meta name="viewport" content="width=device-width,initial-scale=1" />
42 <title>${e} \u2014\xA0${ae.default.message[e]}</title>
43 <link rel="icon" type="image/png" href="https://presta.run/favicon.png">
44 <link rel="icon" type="image/svg" href="https://presta.run/favicon.svg">
45 <link rel="preconnect" href="https://fonts.googleapis.com">
46 <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
47 <link href="https://fonts.googleapis.com/css2?family=Inter:wght@400;900&display=swap" rel="stylesheet">
48 <link rel='stylesheet' href='https://unpkg.com/svbstrate@5.1.0/svbstrate.css' />
49 <style>
50 html,body {
51 font-family: 'Inter', 'sans-serif';
52 color: #23283D;
53 background-color: #DADEF0;
54 }
55 #favicon {
56 fill: #23283D;
57 }
58 @media (prefers-color-scheme: dark) {
59 html,body {
60 color: #DADEF0;
61 background-color: #23283D;
62 }
63 #favicon {
64 fill: #DADEF0;
65 }
66 }
67 </style>
68 </head>
69 <body class='w f aic jcc' style='height: 100vh'>
70 <div class='p12 tac'>
71 <h1>${e}</h1>
72 <p class='mb1'>${ae.default.message[e]}</p>
73
74 <svg width="32" height="32" viewBox="0 0 32 32" fill="none" xmlns="http://www.w3.org/2000/svg">
75 <g clip-path="url(#a)">
76 <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"/>
77 </g>
78 <defs>
79 <clipPath id="a">
80 <path fill="#fff" d="M0 0h32v32H0z"/>
81 </clipPath>
82 </defs>
83 </svg>
84 </div>
85 </body>
86 </html>
87 `}var _e=c(require("url")),We=c(require("query-string")),qe=c(require("raw-body")),Ne=c(require("mime-types")),Rt=/image|audio|video|application\/pdf|application\/zip|applicaton\/octet-stream/i;function Dt(e){return Boolean(e)&&Rt.test(e)}function Ot(e){let t={},r={};for(let o of Object.keys(e)){let n=o.toLowerCase(),i=e[o];!i||(Array.isArray(i)?r[n]=i:t[n]=i)}return{headers:t,multiValueHeaders:r}}function $t(e){let t=(0,We.parse)(e,{arrayFormat:"comma"}),r={},o={};for(let n of Object.keys(t)){let i=t[n];Array.isArray(i)?o[n]=i:r[n]=i||void 0}return{queryStringParameters:r,multiValueQueryStringParameters:o}}async function Le(e){let{url:t="",method:r}=e,{headers:o,multiValueHeaders:n}=Ot(e.headers),i=Dt(o["content-type"]||""),l=o["content-length"],s=l?await(0,qe.default)(e,{limit:"1mb",encoding:Ne.default.charset(l)||void 0}):void 0,a=(0,_e.parse)(t).query||"",{queryStringParameters:p,multiValueQueryStringParameters:m}=$t(a);return o["client-ip"]="0.0.0.0",{rawUrl:t,path:t,httpMethod:r,headers:o,multiValueHeaders:n,rawQuery:a,queryStringParameters:p,multiValueQueryStringParameters:m,body:s?Buffer.from(s).toString(i?"base64":"utf8"):null,isBase64Encoded:i}}function Y(e,t){let r=C(t);for(let o in t.multiValueHeaders)e.setHeader(o,String(t.multiValueHeaders[o]));for(let o in t.headers)e.setHeader(o,String(t.headers[o]));e.statusCode=r.statusCode,e.write(r.body),e.end()}function It({port:e,config:t}){let r=J({port:e}),o=t.staticOutputDir,n=t.assets;return async function(l,s){let a=P(),p=l.url;u({label:"debug",message:`handling ${p}`}),(0,le.default)(n,{dev:!0})(l,s,()=>{u({label:"debug",message:`attempting to serve generated static asset ${p}`}),(0,le.default)(o,{dev:!0})(l,s,async()=>{let m=await Le(l),k=m.headers.Accept||m.headers.accept,g=k&&k.includes("json");try{delete require.cache[t.functionsManifest];let I=require(t.functionsManifest),H=Object.keys(I).map(v=>({matcher:(0,Qe.default)(v),route:v})).filter(({matcher:v})=>v.pattern.test(p.split("?")[0])).map(({route:v})=>I[v])[0];if(H){u({label:"debug",message:`attempting to render lambda for ${p}`});let{handler:v}=require(H),x;try{x=C(await v(m,{}))}catch(Xe){d({label:"serve",message:"lambda",error:Xe}),x=C({statusCode:500,html:g?void 0:X({statusCode:500}),json:g?{detail:ee.default.message[500]}:void 0})}let B=x.headers||{},re=x.statusCode>299&&x.statusCode<399,me=B["Content-Type"],Ke=me?Ge.default.extension(me):"html";y({label:"serve",message:`${x.statusCode} ${re?B.Location:p}`,duration:a()}),Y(s,R(f({},x),{body:Ke==="html"?(x.body||"").split("</body>")[0]+r:x.body}))}else O({label:"serve",message:`404 ${p}`,duration:a()}),Y(s,C({statusCode:404,html:g?void 0:X({statusCode:404})+r,json:g?{detail:ee.default.message[404]}:void 0}))}catch(I){d({label:"serve",message:`500 ${p}`,error:I,duration:a()}),Y(s,C({statusCode:500,html:g?void 0:X({statusCode:500})+r,json:g?{detail:ee.default.message[500]}:void 0}))}})})}}async function ce(e){let t=await(0,Ve.default)({port:e.port}),r=Ue.default.createServer(It({port:t,config:e})).listen(t),o=require("pocket.io")(r,{serveClient:!1});return e.hooks.onBrowserRefresh(()=>{u({label:"debug",message:"refresh event received"}),o.emit("refresh")}),ze.default.watch(e.assets,{ignoreInitial:!0}).on("all",()=>{e.hooks.emitBrowserRefresh()}),{port:t}}var N=(0,Ze.default)("presta"),te="presta.config.js";function pe(e={}){require("module-alias").addAliases({"@":process.cwd(),"presta:internal":__dirname}),require("esbuild-register/dist/node").register(e)}N.version(he.version).option("--config, -c",`Path to a config file. (default ${te})`).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");N.command("build","Build project to output directory.",{default:!0}).example("").example("files/**/*.js").example(`-c ${te}`).action(async e=>{pe(),console.clear();let t=await $({env:b.PRODUCTION,config:j(e.config,!0),cli:R(f({},e),{files:e._})});Je.default.emptyDirSync(t.output),z(`${D.default.blue("presta build")}`),F(),await Me(t)});N.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 ${te}`).action(async e=>{pe(),console.clear();let t=await $({env:b.DEVELOPMENT,config:j(e.config),cli:R(f({},e),{files:e._})});if(e.n)y({label:"dev"}),F();else{let r=await ce(t);z(`${D.default.blue("presta dev")} - http://localhost:${r.port}`),F()}Be(t)});N.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 ${te}`).action(async e=>{pe(),console.clear();let t=await $({env:b.PRODUCTION,config:j(e.config),cli:e}),r=await ce(t);z(`${D.default.blue("presta serve")} - http://localhost:${r.port}`),F()});N.parse(process.argv);
88//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["lib/cli.ts", "lib/log.ts", "lib/constants.ts", "lib/currentPrestaInstance.ts", "lib/config.ts", "lib/createEmitter.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/builtStaticFiles.ts", "lib/removeBuiltStaticFile.ts", "lib/liveReloadScript.ts", "lib/build.ts", "lib/serve.ts", "lib/createDefaultHtmlResponse.ts", "lib/requestToEvent.ts", "lib/sendServerlessResponse.ts"],
  "sourcesContent": ["#!/usr/bin/env node\n\nimport fs from 'fs-extra'\nimport sade from 'sade'\n\nimport pkg from '../package.json'\n\nimport * as logger from './log'\nimport { createConfig, getConfigFile } from './config'\nimport { watch } from './watch'\nimport { build } from './build'\nimport { serve } from './serve'\nimport { Env } from './constants'\n\nconst prog = sade('presta')\nconst CONFIG_DEFAULT = 'presta.config.js'\n\nfunction registerRuntime(options = {}) {\n  require('module-alias').addAliases({\n    '@': process.cwd(),\n    'presta:internal': __dirname, // wherever this is running from\n  })\n\n  require('esbuild-register/dist/node').register(options)\n}\n\nprog\n  .version(pkg.version)\n  // do not provide default config here\n  .option('--config, -c', `Path to a config file.  (default ${CONFIG_DEFAULT})`)\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\nprog\n  .command('build', 'Build project to output directory.', { default: true })\n  .example(``)\n  .example(`files/**/*.js`)\n  .example(`-c ${CONFIG_DEFAULT}`)\n  .action(async (opts) => {\n    registerRuntime()\n\n    console.clear()\n\n    const config = await createConfig({\n      env: Env.PRODUCTION,\n      config: getConfigFile(opts.config, true),\n      cli: {\n        ...opts,\n        files: opts._,\n      },\n    })\n\n    fs.emptyDirSync(config.output)\n\n    logger.raw(`${logger.colors.blue('presta build')}`)\n    logger.newline()\n\n    await build(config)\n  })\n\nprog\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 ${CONFIG_DEFAULT}`)\n  .action(async (opts) => {\n    registerRuntime()\n\n    console.clear()\n\n    const config = await createConfig({\n      env: Env.DEVELOPMENT,\n      config: getConfigFile(opts.config),\n      cli: {\n        ...opts,\n        files: opts._,\n      },\n    })\n\n    if (!opts.n) {\n      const server = await serve(config)\n\n      logger.raw(`${logger.colors.blue('presta dev')} - http://localhost:${server.port}`)\n      logger.newline()\n    } else {\n      logger.info({\n        label: 'dev',\n      })\n      logger.newline()\n    }\n\n    watch(config)\n  })\n\nprog\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 ${CONFIG_DEFAULT}`)\n  .action(async (opts) => {\n    registerRuntime()\n\n    console.clear()\n\n    const config = await createConfig({\n      env: Env.PRODUCTION,\n      config: getConfigFile(opts.config),\n      cli: opts,\n    })\n    const server = await serve(config)\n\n    logger.raw(`${logger.colors.blue('presta serve')} - http://localhost:${server.port}`)\n    logger.newline()\n  })\n\nprog.parse(process.argv)\n", "import c, { Kleur } from 'kleur'\n\nimport { getCurrentPrestaInstance } from './currentPrestaInstance'\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 = getCurrentPrestaInstance().debug\n    const context = getCurrentPrestaInstance().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 { Presta } from './types'\nimport { Env } from './constants'\n\nconst defaultConfig = {\n  pid: process.pid,\n  cwd: process.cwd(),\n  env: Env.PRODUCTION,\n  debug: false,\n} as Presta\n\nexport function setCurrentPrestaInstance(config: Presta): Presta {\n  // @ts-ignore\n  global.__presta__ = config\n  return config\n}\n\nexport function getCurrentPrestaInstance(): Presta {\n  // @ts-ignore\n  if (!global.__presta__) {\n    setCurrentPrestaInstance(defaultConfig)\n  }\n\n  // @ts-ignore\n  return global.__presta__\n}\n", "import fs from 'fs'\nimport path from 'path'\n\nimport * as logger from './log'\nimport { createEmitter, createEmitHook, createOnHook } from './createEmitter'\nimport { setCurrentPrestaInstance, getCurrentPrestaInstance } from './currentPrestaInstance'\nimport { Presta, Config, CLI, Callable } from './types'\nimport { Env } from './constants'\n\nconst defaultConfigFilepath = 'presta.config.js'\n\nfunction resolveAbsolutePaths(\n  config: {\n    files?: string | string[]\n    output?: string\n    assets?: string\n  },\n  { cwd }: { cwd: string }\n) {\n  if (config.files) config.files = ([] as string[]).concat(config.files).map((p) => path.resolve(cwd, p))\n  if (config.output) config.output = path.resolve(cwd, config.output)\n  if (config.assets) config.assets = path.resolve(cwd, config.assets)\n  return config\n}\n\n/**\n * @private\n */\nexport function _clearCurrentConfig() {\n  // @ts-ignore\n  global.__presta__ = {\n    pid: process.pid,\n    cwd: process.cwd(),\n    env: Env.PRODUCTION,\n  }\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    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\n/**\n * Creates a new instance _without_ any values provided by the config file.\n * This is used when the user deletes their config file.\n */\nexport async function removeConfigValues() {\n  logger.debug({\n    label: 'debug',\n    message: `config file values cleared`,\n  })\n\n  return setCurrentPrestaInstance(\n    await createConfig({\n      ...getCurrentPrestaInstance(),\n      config: {},\n    })\n  )\n}\n\nexport async function createConfig({\n  cwd = process.cwd(),\n  env = getCurrentPrestaInstance().env,\n  config = {},\n  cli = {},\n}: {\n  cwd?: string\n  env?: string\n  config?: Partial<Config>\n  cli?: Partial<CLI>\n}) {\n  config = resolveAbsolutePaths({ ...config }, { cwd }) // clone read-only obj\n  cli = resolveAbsolutePaths({ ...cli }, { cwd })\n\n  // combined config, preference to CLI args\n  const merged = {\n    output: path.resolve(cwd, cli.output || config.output || 'build'),\n    assets: path.resolve(cli.assets || config.assets || 'public'),\n    files: cli.files && cli.files.length ? cli.files : config.files ? ([] as string[]).concat(config.files) : [],\n  }\n  const port = cli.port ? parseInt(cli.port) : config.port || 4000\n\n  const previous = getCurrentPrestaInstance()\n  // only create once\n  const emitter = previous.events || createEmitter()\n\n  // deregister old events\n  emitter.clear()\n\n  // set instance\n  const next: Presta = setCurrentPrestaInstance({\n    ...previous,\n    ...merged, // overwrites every time\n    env,\n    cwd,\n    port,\n    debug: cli.debug || getCurrentPrestaInstance().debug,\n    configFilepath: path.resolve(cli.config || defaultConfigFilepath),\n    staticOutputDir: path.join(merged.output, 'static'),\n    functionsOutputDir: path.join(merged.output, 'functions'),\n    functionsManifest: path.join(merged.output, 'routes.json'),\n    events: emitter,\n    hooks: {\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\n  if (config.plugins) {\n    await Promise.all(\n      config.plugins.map((p) => {\n        try {\n          return p(getCurrentPrestaInstance)\n        } catch (e) {\n          logger.error({\n            label: 'error',\n            error: e as Error,\n          })\n        }\n      })\n    )\n  }\n\n  logger.debug({\n    label: 'debug',\n    message: `config created ${JSON.stringify(next)}`,\n  })\n\n  return next\n}\n", "import { Callable } from './types'\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 createEmitHook(name: string, emitter: ReturnType<typeof createEmitter>) {\n  return function hook<T>(props: T) {\n    emitter.emit(name, props)\n  }\n}\n\nexport function createOnHook(name: string, emitter: ReturnType<typeof createEmitter>) {\n  return function hook(callback: Callable) {\n    return emitter.on(name, callback)\n  }\n}\n", "import fs from 'fs-extra'\nimport { create } from 'watch-dependency-graph'\nimport chokidar from 'chokidar'\nimport match from 'picomatch'\n\nimport { outputLambdas } from './outputLambdas'\nimport * as logger from './log'\nimport { getFiles, isStatic, isDynamic } from './getFiles'\nimport { renderStaticEntries } from './renderStaticEntries'\nimport { timer } from './timer'\nimport { createConfig, removeConfigValues, getConfigFile } from './config'\nimport { builtStaticFiles } from './builtStaticFiles'\nimport { removeBuiltStaticFile } from './removeBuiltStaticFile'\nimport { Presta } from './types'\n\n/*\n * Wraps outputLambdas for logging\n */\nfunction updateLambdas(inputs: string[], config: Presta) {\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 async function watch(config: Presta) {\n  /*\n   * Get files that match static/dynamic patters at startup\n   */\n  let files = getFiles(config)\n  let hasConfigFile = fs.existsSync(config.configFilepath)\n\n  if (!files.length) {\n    logger.warn({\n      label: 'paths',\n      message: 'no files configured',\n    })\n  }\n\n  /*\n   * Create initial dynamic entry regardless of if the user has routes, bc we\n   * need this file to serve 404 locally\n   */\n  updateLambdas(files.filter(isDynamic), config)\n\n  /*\n   * Set up all watchers\n   */\n  const fileWatcher = create({ alias: { '@': config.cwd } })\n  const globalWatcher = chokidar.watch(config.cwd, {\n    ignoreInitial: true,\n    ignored: [config.output, config.assets],\n  })\n\n  /*\n   * On a config update, the user may have passed in a new `files` array or\n   * other global config required by all files, so we need to re-fetch all\n   * files and rebuild everything.\n   */\n  async function handleConfigUpdate() {\n    files = getFiles(config)\n    await renderStaticEntries(files.filter(isStatic), config)\n    updateLambdas(files.filter(isDynamic), config)\n  }\n\n  /*\n   * On a changed file, we can just render it\n   */\n  async function handleFileChange(file: string) {\n    // render just file that changed\n    if (isStatic(file)) {\n      await renderStaticEntries([file], config)\n    }\n\n    // update dynamic entry with ALL dynamic files\n    if (isDynamic(file)) {\n      updateLambdas(files.filter(isDynamic), config)\n    }\n\n    config.hooks.emitBrowserRefresh()\n  }\n\n  config.hooks.onBuildFile(({ file }) => {\n    handleFileChange(file)\n  })\n\n  fileWatcher.onRemove(async ([id]) => {\n    logger.debug({\n      label: 'watch',\n      message: `fileWatcher - removed ${id}`,\n    })\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\n    // if it was config, we gotta do a restart\n    if (id === config.configFilepath) {\n      // filter out values from the config file\n      config = await removeConfigValues()\n\n      // reset this!\n      hasConfigFile = false\n\n      handleConfigUpdate()\n    }\n\n    ;(builtStaticFiles[id] || []).forEach((file) => removeBuiltStaticFile(file, config))\n  })\n\n  fileWatcher.onChange(async (files) => {\n    for (const id of files) {\n      logger.debug({\n        label: 'watch',\n        message: `fileWatcher - changed ${id}`,\n      })\n\n      if (id === config.configFilepath) {\n        // clear config file for re-require\n        delete require.cache[config.configFilepath]\n\n        try {\n          // merge in new values from config file\n          config = await createConfig({\n            config: getConfigFile(config.configFilepath),\n          })\n\n          handleConfigUpdate()\n        } catch (e) {\n          logger.error({\n            label: 'error',\n            error: e as Error,\n          })\n        }\n      } else {\n        handleFileChange(id)\n      }\n    }\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  /*\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. It also watches\n   * for addition of a config file.\n   */\n  globalWatcher.on('all', async (event, file) => {\n    // ignore events handled by wdg, or any directory events\n    if (!/add|change/.test(event) || !fs.existsSync(file) || fs.lstatSync(file).isDirectory()) return\n\n    // if a file change matches any pages globs\n    if (match(config.files)(file) && !files.includes(file)) {\n      logger.debug({\n        label: 'watch',\n        message: `globalWatcher - add ${file}`,\n      })\n\n      files.push(file)\n\n      await fileWatcher.add(file)\n\n      handleFileChange(file)\n    }\n\n    // if file matches config file and we don't already have one\n    if (file === config.configFilepath && !hasConfigFile) {\n      logger.debug({\n        label: 'watch',\n        message: `globalWatcher - add config file ${file}`,\n      })\n\n      await fileWatcher.add(config.configFilepath)\n\n      try {\n        // merge in new values from config file\n        config = await createConfig({\n          config: getConfigFile(config.configFilepath),\n        })\n\n        hasConfigFile = true\n\n        handleConfigUpdate()\n      } catch (e) {\n        logger.error({\n          label: 'error',\n          error: e as Error,\n        })\n      }\n    }\n  })\n\n  /**\n   * Init watching after event subscriptions\n   */\n  await fileWatcher.add(files)\n  if (hasConfigFile) await fileWatcher.add(config.configFilepath)\n\n  /**\n   * Prime files to check for errors on startup and register any plugins\n   */\n  try {\n    files.map(require)\n  } catch (e) {\n    logger.error({\n      label: 'error',\n      error: e as Error,\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 { Presta } from './types'\nimport { Env } from './constants'\n\nexport function outputLambda(input: string, config: Presta): [string, string] {\n  const { route } = require(input)\n  const name = path.basename(input).split('.').reverse().slice(1).reverse().join('.')\n  const output = path.join(\n    config.functionsOutputDir,\n    config.env === Env.PRODUCTION ? name + '-' + hashContent(fs.readFileSync(input, 'utf8')) + '.js' : 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';\nimport * as file from '${input}';\nexport const route = file.route\nexport const handler = wrapHandler(file)`\n  )\n\n  return [route, output]\n}\n\nexport function outputLambdas(inputs: string[], config: Presta) {\n  const lambdas = inputs\n    .map((input) => {\n      try {\n        return outputLambda(input, config)\n      } catch (e) {\n        logger.error({\n          label: 'error',\n          error: e as Error,\n        })\n        return null\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'\nimport { Presta } from './types'\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(config: Presta): string[] {\n  try {\n    return ([] as string[])\n      .concat(config.files)\n      .map((file) => globSync(file, { cwd: config.cwd }))\n      .flat()\n      .map((file) => path.resolve(config.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 { builtStaticFiles } from './builtStaticFiles'\nimport { removeBuiltStaticFile } from './removeBuiltStaticFile'\nimport { createLiveReloadScript } from './liveReloadScript'\nimport { Env } from './constants'\nimport { Presta } from './types'\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 function renderStaticEntries(entries: string[], config: Presta): Promise<{ allGeneratedFiles: string[] }> {\n  return new Promise(async (y, n) => {\n    logger.debug({\n      label: 'debug',\n      message: `rendering ${JSON.stringify(entries)}`,\n    })\n\n    const allGeneratedFiles: string[] = []\n    const devClient = createLiveReloadScript({ port: config.port })\n\n    for (const entry of entries) {\n      const location = entry.replace(config.cwd, '')\n\n      try {\n        delete require.cache[entry]\n\n        const file = require(entry)\n        const paths = await file.getStaticPaths()\n\n        const prevFiles = (builtStaticFiles[entry] = builtStaticFiles[entry] || [])\n        const nextFiles: string[] = []\n\n        if (!paths || !paths.length) {\n          logger.warn({\n            label: 'paths',\n            message: `${location} - no paths to render`,\n          })\n\n          prevFiles.forEach((file) => removeBuiltStaticFile(file, config))\n\n          continue\n        }\n\n        for (const url of paths) {\n          const time = timer()\n          const event = {\n            path: url,\n            routeParameters: file.route ? getRouteParams(url, file.route) : {},\n          }\n\n          const response = normalizeResponse(await file.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 + (config.env === Env.PRODUCTION ? '' : devClient)\n\n          fs.outputFileSync(path.join(config.staticOutputDir, filename), html, 'utf-8')\n\n          allGeneratedFiles.push(filename)\n          nextFiles.push(filename)\n\n          logger.info({\n            label: 'built',\n            message: url,\n            duration: time(),\n          })\n        }\n\n        // diff and remove files\n        for (const file of prevFiles) {\n          if (!nextFiles.includes(file)) {\n            removeBuiltStaticFile(file, config)\n          }\n        }\n\n        builtStaticFiles[entry] = nextFiles\n      } catch (e) {\n        if (config.env === 'development') {\n          logger.error({\n            label: 'error',\n            message: 'errors detected, pausing...',\n            error: e as Error,\n          })\n\n          y({ allGeneratedFiles })\n        } else {\n          logger.error({\n            label: 'error',\n            error: e as Error,\n          })\n\n          n(e)\n        }\n\n        // exit loop on any error\n        break\n      }\n    }\n\n    // clear to prevent memory leak\n    // loadCache.clearAllMemory() // TODO probs can't \u2014 emit?\n\n    y({ allGeneratedFiles })\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 { RouteParameters } from './types'\n\n// @see https://github.com/lukeed/regexparam#usage\nexport function getRouteParams(url: string, route: string): RouteParameters {\n  const [path] = url.split('?')\n  const result = toRegExp(route)\n  let i = 0\n  let out: RouteParameters = {}\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", "import { AWS, Response } from './types'\n\nfunction stringify(obj: object | string) {\n  return typeof obj === 'object' ? JSON.stringify(obj) : obj\n}\n\nexport function normalizeResponse(response: Partial<Response> | string): AWS['HandlerResponse'] {\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", "type BuiltStaticFile = string\n\nexport const builtStaticFiles: { [entryFile: string]: BuiltStaticFile[] } = {}\n", "import fs from 'fs-extra'\nimport path from 'path'\n\nimport * as logger from './log'\nimport { Presta } from './types'\n\nexport function removeBuiltStaticFile(file: string, config: Presta) {\n  logger.debug({\n    label: 'debug',\n    message: `removing old static file ${file}`,\n  })\n\n  fs.remove(path.join(config.staticOutputDir, file))\n}\n", "export function createLiveReloadScript({ port }: { port: number }) {\n  return `\n    <script>\n      (function (global) {\n        try {\n          const socketio = document.createElement('script')\n          socketio.src = 'https://unpkg.com/pocket.io@0.1.4/min.js'\n          socketio.onload = function init () {\n            var disconnected = false\n            var socket = io('http://localhost:${port}', {\n              reconnectionAttempts: 3\n            })\n            socket.on('connect', function() { console.log('presta connected on port ${port}') })\n            socket.on('refresh', function() {\n              global.location.reload()\n            })\n            socket.on('disconnect', function() {\n              disconnected = true\n            })\n            socket.on('reconnect_failed', function(e) {\n              if (disconnected) return\n              console.error(\"presta - connection to server on :${port} failed\")\n            })\n          }\n          document.head.appendChild(socketio)\n        } catch (e) {}\n      })(this);\n    </script>\n  `\n}\n", "import fs from 'fs-extra'\nimport { build as esbuild } from 'esbuild'\n\nimport { outputLambdas } from './outputLambdas'\nimport { getFiles, isStatic, isDynamic } from './getFiles'\nimport { renderStaticEntries } from './renderStaticEntries'\nimport { timer } from './timer'\nimport * as logger from './log'\nimport { Presta } from './types'\n\nfunction getRoutesManifestSafely(manifestFilepath: string) {\n  try {\n    return require(manifestFilepath)\n  } catch (e) {\n    return {}\n  }\n}\n\nexport async function build(config: Presta) {\n  const totalTime = timer()\n  const files = getFiles(config)\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 { allGeneratedFiles } = await renderStaticEntries(staticIds, config)\n\n          staticTime = time()\n          staticFileAmount = allGeneratedFiles.length\n        }\n      })(),\n      (async () => {\n        if (dynamicIds.length) {\n          const time = timer()\n\n          outputLambdas(dynamicIds, config)\n\n          await esbuild({\n            entryPoints: Object.values(require(config.functionsManifest)),\n            outdir: config.functionsOutputDir,\n            bundle: true,\n            platform: 'node',\n            target: ['node12'],\n            minify: true,\n            allowOverwrite: 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          logger.error({\n            label: 'error',\n            error: task.reason,\n          })\n        }\n      })\n\n      process.exit(1)\n      return\n    }\n\n    if (staticTime || dynamicTime) {\n      logger.newline()\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    config.hooks.emitPostBuild({\n      output: config.output,\n      staticOutput: config.staticOutputDir,\n      functionsOutput: config.functionsOutputDir,\n      functionsManifest: getRoutesManifestSafely(config.functionsManifest),\n    })\n\n    if (staticTime || dynamicTime) {\n      logger.newline()\n      logger.info({\n        label: 'complete',\n        message: `in ${totalTime()}`,\n      })\n      logger.newline()\n    }\n  }\n}\n", "import fs from 'fs'\nimport path from 'path'\nimport http from 'http'\nimport getPort from 'get-port'\nimport sirv from 'sirv'\nimport chokidar from 'chokidar'\nimport mime from 'mime-types'\nimport toRegExp from 'regexparam'\nimport status from 'statuses'\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 { AWS, Presta } from './types'\nimport { normalizeResponse } from './normalizeResponse'\n\nfunction resolveHTML(dir: string, url: string) {\n  let file = path.join(dir, url)\n\n  // if no extension, it's probably intended to be an HTML file\n  if (!path.extname(url)) {\n    try {\n      return fs.readFileSync(path.join(dir, url, 'index.html'), 'utf8')\n    } catch (e) {}\n  }\n\n  return fs.readFileSync(file, 'utf8')\n}\n\nexport function createServerHandler({ port, config }: { port: number; config: Presta }) {\n  const devClient = createLiveReloadScript({ port })\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    /*\n     * first check the vcs-tracked static folder,\n     * then check the presta-built static folder\n     *\n     * @see https://github.com/sure-thing/presta/issues/30\n     */\n    sirv(assetDir, { dev: true })(req, res, () => {\n      logger.debug({\n        label: 'debug',\n        message: `attempting to serve generated static asset ${url}`,\n      })\n\n      sirv(staticDir, { dev: true })(req, res, async () => {\n        const event = await requestToEvent(req) // stock AWS Event shape\n        const accept = event.headers.Accept || event.headers.accept\n        const acceptsJson = accept && accept.includes('json')\n\n        try {\n          /*\n           * No asset file, no static file, try dynamic\n           */\n          delete require.cache[config.functionsManifest]\n          const manifest = require(config.functionsManifest)\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          /**\n           * If we have a serverless function, delegate to it, otherwise 404\n           */\n          if (lambdaFilepath) {\n            logger.debug({\n              label: 'debug',\n              message: `attempting to render lambda for ${url}`,\n            })\n\n            const { handler }: { handler: AWS['Handler'] } = require(lambdaFilepath)\n            let response: AWS['HandlerResponse']\n\n            try {\n              response = normalizeResponse(await handler(event, {})) // wrapped in ./wrapHandler.ts\n            } catch (e) {\n              logger.error({\n                label: 'serve',\n                message: `lambda`,\n                error: e as Error,\n              })\n\n              response = normalizeResponse({\n                statusCode: 500,\n                html: acceptsJson ? undefined : createDefaultHtmlResponse({ statusCode: 500 }),\n                json: acceptsJson ? { detail: status.message[500] } : undefined,\n              })\n            }\n\n            const headers = response.headers || {}\n            const redir = response.statusCode > 299 && response.statusCode < 399\n\n            // get mime type\n            const type = headers['Content-Type'] as string\n            const ext = type ? mime.extension(type) : 'html'\n\n            logger.info({\n              label: 'serve',\n              message: `${response.statusCode} ${redir ? headers.Location : url}`,\n              duration: time(),\n            })\n\n            sendServerlessResponse(res, {\n              ...response,\n              // only html can be live-reloaded, duh\n              body: ext === 'html' ? (response.body || '').split('</body>')[0] + devClient : response.body,\n            })\n          } else {\n            logger.warn({\n              label: 'serve',\n              message: `404 ${url}`,\n              duration: time(),\n            })\n\n            sendServerlessResponse(\n              res,\n              normalizeResponse({\n                statusCode: 404,\n                html: acceptsJson ? undefined : createDefaultHtmlResponse({ statusCode: 404 }) + devClient,\n                json: acceptsJson ? { detail: status.message[404] } : undefined,\n              })\n            )\n          }\n        } catch (e) {\n          logger.error({\n            label: 'serve',\n            message: `500 ${url}`,\n            error: e as Error,\n            duration: time(),\n          })\n\n          sendServerlessResponse(\n            res,\n            normalizeResponse({\n              statusCode: 500,\n              html: acceptsJson ? undefined : createDefaultHtmlResponse({ statusCode: 500 }) + devClient,\n              json: acceptsJson ? { detail: status.message[500] } : undefined,\n            })\n          )\n        }\n      })\n    })\n  }\n}\n\nexport async function serve(config: Presta) {\n  const port = await getPort({ port: config.port })\n  const server = http.createServer(createServerHandler({ port, config })).listen(port)\n  const socket = require('pocket.io')(server, { serveClient: false })\n\n  config.hooks.onBrowserRefresh(() => {\n    logger.debug({\n      label: 'debug',\n      message: `refresh event received`,\n    })\n\n    socket.emit('refresh')\n  })\n\n  chokidar.watch(config.assets, { ignoreInitial: true }).on('all', () => {\n    config.hooks.emitBrowserRefresh()\n  })\n\n  return { port }\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 { parse as parseQuery } from 'query-string'\nimport rawBody from 'raw-body'\nimport mime from 'mime-types'\nimport { AWS } from './types'\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 function normalizeHeaders(rawHeaders: http.IncomingMessage['headers']) {\n  const headers: AWS['HandlerEvent']['headers'] = {}\n  const multiValueHeaders: AWS['HandlerEvent']['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\nexport function getQueryStringParameters(query: string) {\n  const params = parseQuery(query, { arrayFormat: 'comma' })\n\n  const queryStringParameters: AWS['HandlerEvent']['queryStringParameters'] = {}\n  const multiValueQueryStringParameters: AWS['HandlerEvent']['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 {\n      queryStringParameters[param] = value || undefined\n    }\n  }\n\n  return { queryStringParameters, multiValueQueryStringParameters }\n}\n\nexport async function requestToEvent(req: http.IncomingMessage): Promise<AWS['HandlerEvent']> {\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: mime.charset(contentLengthHeader) || undefined,\n      })\n    : undefined\n  const rawQuery = parseUrl(path).query || ''\n  const { queryStringParameters, multiValueQueryStringParameters } = getQueryStringParameters(rawQuery)\n\n  /**\n   * Just fake this locally\n   * @see https://github.com/netlify/cli/blob/27bb7b9b30d465abe86f87f4274dd7a71b1b003b/src/utils/serve-functions.js#L208\n   */\n  headers['client-ip'] = '0.0.0.0'\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  }\n}\n", "import http from 'http'\n\nimport { normalizeResponse } from './normalizeResponse'\nimport { AWS } from './types'\n\nexport function sendServerlessResponse(res: http.ServerResponse, r: Partial<AWS['HandlerResponse']>) {\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": []
}
