klaudius 0.5.0 → 0.5.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -55,4 +55,4 @@ Email [hello@klaudius.dev](mailto:hello@klaudius.dev).
55
55
 
56
56
  ## About
57
57
 
58
- Klaudius is operated by Cloudbot Limited, a company registered in England and Wales. More at [klaudius.dev](https://klaudius.dev).
58
+ More at [klaudius.dev](https://klaudius.dev).
package/dist/bin.js CHANGED
@@ -8,9 +8,9 @@ Please share that file so we can trace what happened.`);return t}function C(e){l
8
8
  `,l=["","# "+"=".repeat(60),"# Backfilled by `klaudius update` (new template env keys)","# "+"=".repeat(60)];for(let c of t)l.push(""),c.comment&&l.push(`# ${c.comment}`),l.push(Ce(c.key,c.value));return e+a+l.join(`
9
9
  `)+`
10
10
  `}function ve(e){return e.length>=2&&e.startsWith('"')&&e.endsWith('"')?e.slice(1,-1).replace(/\\\\/g,"\\").replace(/\\"/g,'"').replace(/\\\$/g,"$").replace(/\\`/g,"`"):e.length>=2&&e.startsWith("'")&&e.endsWith("'")?e.slice(1,-1):e}function A(e){let s={};for(let n of e.split(`
11
- `)){let t=n.trim();if(!t||t.startsWith("#"))continue;let a=t.indexOf("=");if(a===-1)continue;let l=t.slice(0,a).trim(),c=ve(t.slice(a+1).trim());s[l]=c}return s}function z(e){let s=l=>{let c=e[l];if(!c)return;let r=parseInt(c,10);return isNaN(r)?void 0:r},n=e.SMS_PROVIDER,t=e.OUTREACH_CHANNELS?.split(",").map(l=>l.trim()).filter(l=>l==="email"||l==="sms"),a=e.OUTREACH_PRIORITY;return{licenseKey:e.KLAUDIUS_LICENSE_KEY,pricing:e.PRICING,pricingTerms:e.PRICING_TERMS,pricingMonthly:e.PRICING_MONTHLY||void 0,outreachEnabled:e.OUTREACH_ENABLED?e.OUTREACH_ENABLED.toLowerCase()==="true":void 0,name:e.OPERATOR_NAME,signature:e.SIGNATURE,outreachChannels:t&&t.length>0?t:void 0,outreachPriority:a==="email"||a==="sms"?a:void 0,emailAddress:e.EMAIL_ADDRESS,emailFromName:e.EMAIL_FROM_NAME,emailSmtpHost:e.EMAIL_SMTP_HOST,emailSmtpPort:s("EMAIL_SMTP_PORT"),emailImapHost:e.EMAIL_IMAP_HOST,emailImapPort:s("EMAIL_IMAP_PORT"),emailSentFolder:e.EMAIL_SENT_FOLDER,emailInboxFolder:e.EMAIL_INBOX_FOLDER,smsProvider:n==="imessage"||n==="twilio"?n:void 0,testPhone:e.TEST_PHONE,twilioAccountSid:e.TWILIO_ACCOUNT_SID,twilioPhoneNumber:e.TWILIO_PHONE_NUMBER,twilioAuthToken:e.TWILIO_AUTH_TOKEN,supabaseUrl:e.SUPABASE_URL,supabaseProjectRef:e.SUPABASE_PROJECT_REF,supabaseServiceKey:e.SUPABASE_SERVICE_KEY,supabasePat:e.SUPABASE_PAT,vercelToken:e.VERCEL_TOKEN,googlePlacesApiKey:e.GOOGLE_PLACES_API_KEY,serpapiKey:e.SERPAPI_KEY,emailPassword:e.EMAIL_PASSWORD,telegramBotToken:e.TELEGRAM_BOT_TOKEN,telegramChatId:e.TELEGRAM_CHAT_ID}}function K(e){let s=u=>{let o=e[u];if(!o)return;let p=parseInt(o,10);return isNaN(p)?void 0:p},n=e.SUPABASE_URL,t=e.SUPABASE_SERVICE_KEY,a=e.SUPABASE_PAT,l=e.SUPABASE_PROJECT_REF,c=e.PRICING,r=e.PRICING_TERMS||"one-off, no monthly fees",h=e.VERCEL_TOKEN;if(!c||!n||!t||!a||!l||!h)return null;let f=e.SMS_PROVIDER,k=e.OUTREACH_CHANNELS?.split(",").map(u=>u.trim()).filter(u=>u==="email"||u==="sms"),g=e.OUTREACH_PRIORITY;return{licenseKey:e.KLAUDIUS_LICENSE_KEY,pricing:c,pricingTerms:r,pricingMonthly:e.PRICING_MONTHLY||void 0,outreachEnabled:e.OUTREACH_ENABLED?.toLowerCase()==="true",name:e.OPERATOR_NAME,signature:e.SIGNATURE,outreachChannels:k&&k.length>0?k:void 0,outreachPriority:g==="email"||g==="sms"?g:void 0,emailAddress:e.EMAIL_ADDRESS,emailPassword:e.EMAIL_PASSWORD,emailFromName:e.EMAIL_FROM_NAME,emailSmtpHost:e.EMAIL_SMTP_HOST,emailSmtpPort:s("EMAIL_SMTP_PORT"),emailImapHost:e.EMAIL_IMAP_HOST,emailImapPort:s("EMAIL_IMAP_PORT"),emailSentFolder:e.EMAIL_SENT_FOLDER,emailInboxFolder:e.EMAIL_INBOX_FOLDER,smsProvider:f==="imessage"||f==="twilio"?f:void 0,testPhone:e.TEST_PHONE,twilioAccountSid:e.TWILIO_ACCOUNT_SID,twilioAuthToken:e.TWILIO_AUTH_TOKEN,twilioPhoneNumber:e.TWILIO_PHONE_NUMBER,vercelToken:h,supabaseUrl:n,supabaseServiceKey:t,supabasePat:a,supabaseProjectRef:l,googlePlacesApiKey:e.GOOGLE_PLACES_API_KEY,serpapiKey:e.SERPAPI_KEY,telegramBotToken:e.TELEGRAM_BOT_TOKEN,telegramChatId:e.TELEGRAM_CHAT_ID}}import{mkdir as Oe,rm as Le}from"fs/promises";import{Readable as Me}from"stream";import*as X from"tar";var xe="https://klaudius.dev/api/template";function $e(e){return Me.fromWeb(e)}async function O(e){let s=process.env.KLAUDIUS_TEMPLATE_URL??xe,n="0.5.0",t;try{t=await fetch(s,{method:"POST",headers:{Authorization:`Bearer ${e.licenseKey}`,"Content-Type":"application/json"},body:JSON.stringify({machine_id:e.machineId,hostname:e.hostname,os:e.os,cli_version:n})})}catch(c){throw new Error(`Could not reach klaudius.dev to download the template: ${c.message}`)}if(!t.ok){let c="";try{let r=await t.json();c=r.reason?` (${r.reason})`:""}catch{}throw new Error(`Template download failed: HTTP ${t.status}${c}`)}if(!t.body)throw new Error("Template download succeeded but response had no body");let a=t.headers.get("x-klaudius-tier");await Oe(e.destDir,{recursive:!0});let l=0;try{await new Promise((c,r)=>{let h=X.extract({cwd:e.destDir,strict:!0,onReadEntry:()=>{l++}});h.on("finish",c),h.on("error",r),$e(t.body).on("error",r).pipe(h)})}catch(c){throw await Le(e.destDir,{recursive:!0,force:!0}).catch(()=>{}),new Error(`Failed to extract template archive: ${c.message}`)}return{filesExtracted:l,tier:a}}import{mkdir as Ne,readFile as Ue,writeFile as Ke,access as De}from"fs/promises";import{dirname as He,join as q}from"path";var je=".klaudius",J=`${je}/manifest.json`;async function Q(e){let s=q(e,J);try{await De(s)}catch{return null}let n=await Ue(s,"utf-8");return JSON.parse(n)}async function L(e,s){let n=q(e,J);await Ne(He(n),{recursive:!0}),await Ke(n,JSON.stringify(s,null,2)+`
11
+ `)){let t=n.trim();if(!t||t.startsWith("#"))continue;let a=t.indexOf("=");if(a===-1)continue;let l=t.slice(0,a).trim(),c=ve(t.slice(a+1).trim());s[l]=c}return s}function z(e){let s=l=>{let c=e[l];if(!c)return;let r=parseInt(c,10);return isNaN(r)?void 0:r},n=e.SMS_PROVIDER,t=e.OUTREACH_CHANNELS?.split(",").map(l=>l.trim()).filter(l=>l==="email"||l==="sms"),a=e.OUTREACH_PRIORITY;return{licenseKey:e.KLAUDIUS_LICENSE_KEY,pricing:e.PRICING,pricingTerms:e.PRICING_TERMS,pricingMonthly:e.PRICING_MONTHLY||void 0,outreachEnabled:e.OUTREACH_ENABLED?e.OUTREACH_ENABLED.toLowerCase()==="true":void 0,name:e.OPERATOR_NAME,signature:e.SIGNATURE,outreachChannels:t&&t.length>0?t:void 0,outreachPriority:a==="email"||a==="sms"?a:void 0,emailAddress:e.EMAIL_ADDRESS,emailFromName:e.EMAIL_FROM_NAME,emailSmtpHost:e.EMAIL_SMTP_HOST,emailSmtpPort:s("EMAIL_SMTP_PORT"),emailImapHost:e.EMAIL_IMAP_HOST,emailImapPort:s("EMAIL_IMAP_PORT"),emailSentFolder:e.EMAIL_SENT_FOLDER,emailInboxFolder:e.EMAIL_INBOX_FOLDER,smsProvider:n==="imessage"||n==="twilio"?n:void 0,testPhone:e.TEST_PHONE,twilioAccountSid:e.TWILIO_ACCOUNT_SID,twilioPhoneNumber:e.TWILIO_PHONE_NUMBER,twilioAuthToken:e.TWILIO_AUTH_TOKEN,supabaseUrl:e.SUPABASE_URL,supabaseProjectRef:e.SUPABASE_PROJECT_REF,supabaseServiceKey:e.SUPABASE_SERVICE_KEY,supabasePat:e.SUPABASE_PAT,vercelToken:e.VERCEL_TOKEN,googlePlacesApiKey:e.GOOGLE_PLACES_API_KEY,serpapiKey:e.SERPAPI_KEY,emailPassword:e.EMAIL_PASSWORD,telegramBotToken:e.TELEGRAM_BOT_TOKEN,telegramChatId:e.TELEGRAM_CHAT_ID}}function K(e){let s=u=>{let o=e[u];if(!o)return;let p=parseInt(o,10);return isNaN(p)?void 0:p},n=e.SUPABASE_URL,t=e.SUPABASE_SERVICE_KEY,a=e.SUPABASE_PAT,l=e.SUPABASE_PROJECT_REF,c=e.PRICING,r=e.PRICING_TERMS||"one-off, no monthly fees",h=e.VERCEL_TOKEN;if(!c||!n||!t||!a||!l||!h)return null;let f=e.SMS_PROVIDER,k=e.OUTREACH_CHANNELS?.split(",").map(u=>u.trim()).filter(u=>u==="email"||u==="sms"),g=e.OUTREACH_PRIORITY;return{licenseKey:e.KLAUDIUS_LICENSE_KEY,pricing:c,pricingTerms:r,pricingMonthly:e.PRICING_MONTHLY||void 0,outreachEnabled:e.OUTREACH_ENABLED?.toLowerCase()==="true",name:e.OPERATOR_NAME,signature:e.SIGNATURE,outreachChannels:k&&k.length>0?k:void 0,outreachPriority:g==="email"||g==="sms"?g:void 0,emailAddress:e.EMAIL_ADDRESS,emailPassword:e.EMAIL_PASSWORD,emailFromName:e.EMAIL_FROM_NAME,emailSmtpHost:e.EMAIL_SMTP_HOST,emailSmtpPort:s("EMAIL_SMTP_PORT"),emailImapHost:e.EMAIL_IMAP_HOST,emailImapPort:s("EMAIL_IMAP_PORT"),emailSentFolder:e.EMAIL_SENT_FOLDER,emailInboxFolder:e.EMAIL_INBOX_FOLDER,smsProvider:f==="imessage"||f==="twilio"?f:void 0,testPhone:e.TEST_PHONE,twilioAccountSid:e.TWILIO_ACCOUNT_SID,twilioAuthToken:e.TWILIO_AUTH_TOKEN,twilioPhoneNumber:e.TWILIO_PHONE_NUMBER,vercelToken:h,supabaseUrl:n,supabaseServiceKey:t,supabasePat:a,supabaseProjectRef:l,googlePlacesApiKey:e.GOOGLE_PLACES_API_KEY,serpapiKey:e.SERPAPI_KEY,telegramBotToken:e.TELEGRAM_BOT_TOKEN,telegramChatId:e.TELEGRAM_CHAT_ID}}import{mkdir as Oe,rm as Le}from"fs/promises";import{Readable as Me}from"stream";import*as X from"tar";var xe="https://klaudius.dev/api/template";function $e(e){return Me.fromWeb(e)}async function O(e){let s=process.env.KLAUDIUS_TEMPLATE_URL??xe,n="0.5.1",t;try{t=await fetch(s,{method:"POST",headers:{Authorization:`Bearer ${e.licenseKey}`,"Content-Type":"application/json"},body:JSON.stringify({machine_id:e.machineId,hostname:e.hostname,os:e.os,cli_version:n})})}catch(c){throw new Error(`Could not reach klaudius.dev to download the template: ${c.message}`)}if(!t.ok){let c="";try{let r=await t.json();c=r.reason?` (${r.reason})`:""}catch{}throw new Error(`Template download failed: HTTP ${t.status}${c}`)}if(!t.body)throw new Error("Template download succeeded but response had no body");let a=t.headers.get("x-klaudius-tier");await Oe(e.destDir,{recursive:!0});let l=0;try{await new Promise((c,r)=>{let h=X.extract({cwd:e.destDir,strict:!0,onReadEntry:()=>{l++}});h.on("finish",c),h.on("error",r),$e(t.body).on("error",r).pipe(h)})}catch(c){throw await Le(e.destDir,{recursive:!0,force:!0}).catch(()=>{}),new Error(`Failed to extract template archive: ${c.message}`)}return{filesExtracted:l,tier:a}}import{mkdir as Ne,readFile as Ue,writeFile as Ke,access as De}from"fs/promises";import{dirname as He,join as q}from"path";var je=".klaudius",J=`${je}/manifest.json`;async function Q(e){let s=q(e,J);try{await De(s)}catch{return null}let n=await Ue(s,"utf-8");return JSON.parse(n)}async function L(e,s){let n=q(e,J);await Ne(He(n),{recursive:!0}),await Ke(n,JSON.stringify(s,null,2)+`
12
12
  `,"utf-8")}function Z(e,s){let n=new Date().toISOString();return{version:1,templateVersion:e,installedAt:n,updatedAt:n,files:s}}import{createHash as Be}from"crypto";import{hostname as ee,userInfo as Fe,platform as te,arch as se}from"os";import E from"picocolors";var i={info:e=>console.log(E.cyan("\u2139"),e),success:e=>console.log(E.green("\u2713"),e),warn:e=>console.log(E.yellow("\u26A0"),e),error:e=>console.error(E.red("\u2717"),e),step:e=>console.log(E.dim("\u2192"),e),blank:()=>console.log(""),heading:e=>console.log(`
13
- `+E.bold(E.cyan(e))),detail:e=>console.log(E.dim(" "+e))};var Ve="https://klaudius.dev",Ge="/api/licenses/validate",We="0.5.0";async function ie(e){if(!e||e.trim().length===0)return{ok:!1,reason:"No license key provided. Pass one via --license <key> or paste it when prompted."};let s=e.trim(),t=(process.env.KLAUDIUS_API_URL??Ve).replace(/\/$/,"")+Ge,a=b(),l;try{l=await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({license_key:s,machine_id:a,hostname:ee(),os:`${te()}-${se()}`,cli_version:We})})}catch(r){return{ok:!1,reason:`Could not reach the Klaudius license server (${r instanceof Error?r.message:"unknown"}). Check your internet connection or try again later.`}}let c;try{c=await l.json()}catch{return{ok:!1,reason:`License server returned an unparseable response (HTTP ${l.status}).`}}return c.ok?{ok:!0,tier:c.tier??"core"}:{ok:!1,reason:c.reason??`License rejected (HTTP ${l.status}).`}}function b(){let e=[ee(),Fe().username,te(),se()].join("|");return Be("sha256").update(e).digest("hex").slice(0,32)}import{readFile as Ye}from"fs/promises";import{join as ze}from"path";async function M(e){try{let s=await fetch(e.url+"/rest/v1/",{headers:{apikey:e.serviceKey,Authorization:`Bearer ${e.serviceKey}`}});return s.status===401||s.status===403?{ok:!1,message:`Supabase rejected auth (HTTP ${s.status}). Check SUPABASE_SERVICE_KEY.`}:{ok:!0}}catch(s){return{ok:!1,message:`Could not reach Supabase: ${s.message}`}}}async function ne(e){let s;try{s=await Ye(ze(e.projectRoot,"scripts","schema.sql"),"utf-8")}catch(n){return{ok:!1,message:`Could not read scripts/schema.sql: ${n.message}`}}try{let n=await fetch(`https://api.supabase.com/v1/projects/${e.projectRef}/database/query`,{method:"POST",headers:{Authorization:`Bearer ${e.pat}`,"Content-Type":"application/json"},body:JSON.stringify({query:s})});if(n.ok)return{ok:!0};let t=await n.text();return t.includes("already exists")||t.includes('relation "clients" already exists')||t.includes("duplicate")?(i.detail("Schema appears already applied (tables exist). Skipping."),{ok:!0}):{ok:!1,message:`Supabase Management API rejected schema (HTTP ${n.status}): ${t.slice(0,300)}`}}catch(n){return{ok:!1,message:`Schema apply failed: ${n.message}`}}}async function re(e){try{let s=await fetch("https://api.vercel.com/v2/user",{headers:{Authorization:`Bearer ${e}`}});if(s.status===401||s.status===403)return{ok:!1,message:`Vercel rejected the token (HTTP ${s.status}). The token is invalid, expired, or has been revoked.`};if(!s.ok)return{ok:!1,message:`Vercel API returned HTTP ${s.status}. Try again or check status.vercel.com.`};let n=await s.json();return{ok:!0,username:n.user?.username??n.user?.email}}catch(s){return{ok:!1,message:`Could not reach Vercel: ${s.message}`}}}async function ae(e){try{let s=await fetch("https://places.googleapis.com/v1/places:searchText",{method:"POST",headers:{"Content-Type":"application/json","X-Goog-Api-Key":e,"X-Goog-FieldMask":"places.id"},body:JSON.stringify({textQuery:"x"})});if(s.ok)return{ok:!0};if(s.status===429)return{ok:!0};let n={};try{n=await s.json()}catch{}let t=n.error?.message??"",a=n.error?.status??"";return s.status===400&&(a==="PERMISSION_DENIED"||/Places API \(New\) has not been enabled|API_KEY_SERVICE_BLOCKED/i.test(t))?{ok:!1,message:"Key is valid, but Places API (New) is not enabled on its Cloud project. Open https://console.cloud.google.com/apis/library and enable 'Places API (New)' for the project this key belongs to, then try again."}:s.status===401||s.status===403?{ok:!1,message:t||`Google rejected the key (HTTP ${s.status}). Common causes: invalid key, billing not enabled on the Cloud project, or API key restrictions blocking this caller.`}:{ok:!1,message:`Google API returned HTTP ${s.status}${t?`: ${t}`:""}.`}}catch(s){return{ok:!1,message:`Could not reach Google: ${s.message}`}}}import Xe from"nodemailer";import{ImapFlow as qe}from"imapflow";async function le(e){let s=Xe.createTransport({host:e.emailSmtpHost,port:e.emailSmtpPort,secure:e.emailSmtpPort===465,auth:{user:e.emailAddress,pass:e.emailPassword},connectionTimeout:1e4,greetingTimeout:5e3,socketTimeout:1e4});try{await s.verify()}catch(t){return{ok:!1,failedAt:"smtp",message:oe(t,"SMTP")}}finally{s.close()}let n=new qe({host:e.emailImapHost,port:e.emailImapPort,secure:e.emailImapPort===993,auth:{user:e.emailAddress,pass:e.emailPassword},logger:!1});try{await n.connect(),await n.logout()}catch(t){try{await n.logout()}catch{}return{ok:!1,failedAt:"imap",message:oe(t,"IMAP")}}return{ok:!0}}function oe(e,s){let n=e instanceof Error?e.message:String(e),t=n.toLowerCase();return t.includes("invalid credentials")||t.includes("authentication failed")||t.includes("auth failed")||t.includes("login failed")?`${s} rejected the credentials. Most common causes:
13
+ `+E.bold(E.cyan(e))),detail:e=>console.log(E.dim(" "+e))};var Ve="https://klaudius.dev",Ge="/api/licenses/validate",We="0.5.1";async function ie(e){if(!e||e.trim().length===0)return{ok:!1,reason:"No license key provided. Pass one via --license <key> or paste it when prompted."};let s=e.trim(),t=(process.env.KLAUDIUS_API_URL??Ve).replace(/\/$/,"")+Ge,a=b(),l;try{l=await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({license_key:s,machine_id:a,hostname:ee(),os:`${te()}-${se()}`,cli_version:We})})}catch(r){return{ok:!1,reason:`Could not reach the Klaudius license server (${r instanceof Error?r.message:"unknown"}). Check your internet connection or try again later.`}}let c;try{c=await l.json()}catch{return{ok:!1,reason:`License server returned an unparseable response (HTTP ${l.status}).`}}return c.ok?{ok:!0,tier:c.tier??"core"}:{ok:!1,reason:c.reason??`License rejected (HTTP ${l.status}).`}}function b(){let e=[ee(),Fe().username,te(),se()].join("|");return Be("sha256").update(e).digest("hex").slice(0,32)}import{readFile as Ye}from"fs/promises";import{join as ze}from"path";async function M(e){try{let s=await fetch(e.url+"/rest/v1/",{headers:{apikey:e.serviceKey,Authorization:`Bearer ${e.serviceKey}`}});return s.status===401||s.status===403?{ok:!1,message:`Supabase rejected auth (HTTP ${s.status}). Check SUPABASE_SERVICE_KEY.`}:{ok:!0}}catch(s){return{ok:!1,message:`Could not reach Supabase: ${s.message}`}}}async function ne(e){let s;try{s=await Ye(ze(e.projectRoot,"scripts","schema.sql"),"utf-8")}catch(n){return{ok:!1,message:`Could not read scripts/schema.sql: ${n.message}`}}try{let n=await fetch(`https://api.supabase.com/v1/projects/${e.projectRef}/database/query`,{method:"POST",headers:{Authorization:`Bearer ${e.pat}`,"Content-Type":"application/json"},body:JSON.stringify({query:s})});if(n.ok)return{ok:!0};let t=await n.text();return t.includes("already exists")||t.includes('relation "clients" already exists')||t.includes("duplicate")?(i.detail("Schema appears already applied (tables exist). Skipping."),{ok:!0}):{ok:!1,message:`Supabase Management API rejected schema (HTTP ${n.status}): ${t.slice(0,300)}`}}catch(n){return{ok:!1,message:`Schema apply failed: ${n.message}`}}}async function re(e){try{let s=await fetch("https://api.vercel.com/v2/user",{headers:{Authorization:`Bearer ${e}`}});if(s.status===401||s.status===403)return{ok:!1,message:`Vercel rejected the token (HTTP ${s.status}). The token is invalid, expired, or has been revoked.`};if(!s.ok)return{ok:!1,message:`Vercel API returned HTTP ${s.status}. Try again or check status.vercel.com.`};let n=await s.json();return{ok:!0,username:n.user?.username??n.user?.email}}catch(s){return{ok:!1,message:`Could not reach Vercel: ${s.message}`}}}async function ae(e){try{let s=await fetch("https://places.googleapis.com/v1/places:searchText",{method:"POST",headers:{"Content-Type":"application/json","X-Goog-Api-Key":e,"X-Goog-FieldMask":"places.id"},body:JSON.stringify({textQuery:"x"})});if(s.ok)return{ok:!0};if(s.status===429)return{ok:!0};let n={};try{n=await s.json()}catch{}let t=n.error?.message??"",a=n.error?.status??"";return s.status===400&&(a==="PERMISSION_DENIED"||/Places API \(New\) has not been enabled|API_KEY_SERVICE_BLOCKED/i.test(t))?{ok:!1,message:"Key is valid, but Places API (New) is not enabled on its Cloud project. Open https://console.cloud.google.com/apis/library and enable 'Places API (New)' for the project this key belongs to, then try again."}:s.status===401||s.status===403?{ok:!1,message:t||`Google rejected the key (HTTP ${s.status}). Common causes: invalid key, billing not enabled on the Cloud project, or API key restrictions blocking this caller.`}:{ok:!1,message:`Google API returned HTTP ${s.status}${t?`: ${t}`:""}.`}}catch(s){return{ok:!1,message:`Could not reach Google: ${s.message}`}}}import Xe from"nodemailer";import{ImapFlow as qe}from"imapflow";async function le(e){let s=Xe.createTransport({host:e.emailSmtpHost,port:e.emailSmtpPort,secure:e.emailSmtpPort===465,auth:{user:e.emailAddress,pass:e.emailPassword},connectionTimeout:1e4,greetingTimeout:5e3,socketTimeout:1e4});try{await s.verify()}catch(t){return{ok:!1,failedAt:"smtp",message:oe(t,"SMTP")}}finally{s.close()}let n=new qe({host:e.emailImapHost,port:e.emailImapPort,secure:e.emailImapPort===993,auth:{user:e.emailAddress,pass:e.emailPassword},logger:!1});try{await n.connect(),await n.logout()}catch(t){try{await n.logout()}catch{}return{ok:!1,failedAt:"imap",message:oe(t,"IMAP")}}return{ok:!0}}function oe(e,s){let n=e instanceof Error?e.message:String(e),t=n.toLowerCase();return t.includes("invalid credentials")||t.includes("authentication failed")||t.includes("auth failed")||t.includes("login failed")?`${s} rejected the credentials. Most common causes:
14
14
  \u2022 The app password is wrong, expired, or for a different provider.
15
15
  \u2022 The email address doesn't match the account that issued the password.
16
16
  \u2022 2FA is required but no app-specific password was generated.`:t.includes("etimedout")||t.includes("timeout")?`${s} server didn't respond in time. Check the host/port, or your network.`:t.includes("enotfound")||t.includes("getaddrinfo")?`${s} host could not be resolved. Check the host setting.`:`${s} check failed: ${n}`}async function ce(e){if(!e.accountSid||!e.authToken)return{ok:!1,message:"Missing Twilio Account SID or Auth Token."};let s=`https://api.twilio.com/2010-04-01/Accounts/${encodeURIComponent(e.accountSid)}.json`,n=Buffer.from(`${e.accountSid}:${e.authToken}`).toString("base64");try{let t=await fetch(s,{headers:{Authorization:`Basic ${n}`}});if(t.status===401)return{ok:!1,message:"Twilio rejected the credentials (401). Check the Account SID and Auth Token from https://console.twilio.com/ \u2014 the Auth Token shown there is the value to paste."};if(!t.ok)return{ok:!1,message:`Twilio API returned HTTP ${t.status}. Try again or check status.twilio.com.`};let a=await t.json();return a.status&&a.status!=="active"?{ok:!1,message:`Twilio account is in '${a.status}' state. Check the console for any holds before sending SMS.`}:{ok:!0,friendlyName:a.friendly_name}}catch(t){return{ok:!1,message:`Could not reach Twilio: ${t.message}`}}}import{spawn as Je}from"child_process";async function ue(e){if(process.platform!=="darwin")return{ok:!1,message:"iMessage send is only supported on macOS. To run init from a non-Mac, switch the SMS provider to Twilio in the wizard."};let n=`tell application "Messages" to send "Klaudius connected (setup ping; ignore)." to buddy "${e.testPhone.replace(/"/g,"")}"`,t=await Qe(n);if(t.code===0)return{ok:!0};let a=t.stderr.toLowerCase();return a.includes("can't get application")||a.includes("isn't running")||a.includes("can't make some data")?{ok:!1,message:"Messages.app didn't respond. Open Messages on your Mac and sign in with your Apple ID, then retry."}:a.includes("(-1728)")||a.includes("not found")||a.includes("buddy")?{ok:!1,message:`Couldn't reach ${e.testPhone}. Either the number format is wrong (use the international format with a leading +) or this Mac doesn't have a path to that number. If TEST_PHONE is a non-Apple phone (Android), you need an iPhone signed into the same Apple ID with Settings \u2192 Messages \u2192 Text Message Forwarding turned ON for this Mac.`}:a.includes("(-1719)")||a.includes("(-25006)")?{ok:!1,message:"Messages.app refused the send \u2014 usually means iMessage isn't configured on this Mac. Open Messages \u2192 Settings \u2192 iMessage and sign in with your Apple ID."}:{ok:!1,message:`osascript error: ${t.stderr.trim()||`exit code ${t.code}`}`}}function Qe(e){return new Promise(s=>{let n=Je("osascript",["-e",e],{stdio:["ignore","ignore","pipe"]}),t="";n.stderr.on("data",a=>{t+=a.toString()}),n.on("close",a=>s({code:a??1,stderr:t})),n.on("error",a=>s({code:1,stderr:a.message}))})}import{spawn as me}from"child_process";function pe(e,s,n){return new Promise(t=>{let a=me(e,s,{cwd:n.cwd,stdio:n.inherit!==!1?"inherit":"pipe",shell:!1});a.on("close",l=>t(l??1)),a.on("error",()=>t(1))})}function de(e,s,n){return new Promise(t=>{let a=me(e,s,{cwd:n.cwd,stdio:["ignore","pipe","pipe"],shell:!1}),l="",c="";a.stdout?.on("data",r=>{l+=r.toString()}),a.stderr?.on("data",r=>{c+=r.toString()}),a.on("close",r=>t({code:r??1,stdout:l,stderr:c})),a.on("error",r=>t({code:1,stdout:l,stderr:c+r.message}))})}async function Ze(e){let s=["supabase","python-dotenv","requests","httpx","Pillow"],n=["install","--quiet","--user"],t=await de("pip3",[...n,"--break-system-packages",...s],{cwd:e});return t.code!==0&&/no such option:.*break-system-packages/i.test(t.stderr)&&(i.detail("Older pip detected; retrying without --break-system-packages"),t=await de("pip3",[...n,...s],{cwd:e})),t.code!==0?(t.stderr.trim()&&process.stderr.write(t.stderr),t.stdout.trim()&&process.stdout.write(t.stdout),"failed"):"ok"}async function fe(e){let s={npm:"skipped",pip:"skipped",playwright:"skipped"};i.step("Running npm install (Next.js, Playwright JS, etc.)...");let n=await pe("npm",["install"],{cwd:e});s.npm=n===0?"ok":"failed",i.step("Installing Python packages (supabase, dotenv, requests, httpx, Pillow)..."),s.pip=await Ze(e),i.step("Installing Playwright browser (chromium ~200MB; one-time)...");let t=await pe("npx",["playwright","install","chromium"],{cwd:e});return s.playwright=t===0?"ok":"failed",s}async function we(e){let s=e.target??".",n=it(s)?s:st(process.cwd(),s),t=await nt(n);if(t.kind==="occupied"){let u=s===".";i.error("Klaudius needs an empty folder to scaffold into. This directory isn't empty."),i.detail(`Path: ${n}`),i.detail(`Contains: ${t.entries.slice(0,5).join(", ")}${t.entries.length>5?`, \u2026 (+${t.entries.length-5} more)`:""}`),i.blank(),u?(i.detail("Two ways forward:"),i.blank(),i.detail("1. Pass a folder name and Klaudius creates it for you:"),console.log(` ${m.cyan("npx klaudius init my-agency")}`),i.blank(),i.detail("2. Or cd into an empty folder first, then re-run:"),console.log(` ${m.cyan("mkdir my-agency && cd my-agency && npx klaudius init")}`)):(i.detail(`The folder '${s}' already exists and isn't empty.`),i.detail("Either pick a different name:"),console.log(` ${m.cyan(`npx klaudius init ${s}-2`)}`),i.detail("\u2026or delete the existing folder if it's safe to remove.")),process.exit(1)}let a=await at(e.license),l=d.spinner();l.start("Validating license");let c=await ie(a);c.ok||(l.stop(`License invalid: ${c.reason??"unknown"}`,2),i.detail("If you believe this is in error, email hello@klaudius.dev with your purchase details."),process.exit(1)),l.stop(`License valid (${c.tier??"core"} tier)`);let r,h=!1;if(t.kind==="partial")if(await rt(n)==="resume"){let o=await he(t.envPath,"utf-8"),p=K(A(o));p?(i.success("Resuming with values from your existing .env"),i.detail("(Skipping the wizard \u2014 edit .env directly if you need to change anything.)"),r=p,h=!0):(i.warn("Existing .env was missing required fields \u2014 starting wizard fresh."),r=await y())}else{let o=await he(t.envPath,"utf-8"),p=K(A(o));r=await y(p??{})}else r=await y();for(r.licenseKey=a,await S(n,r),h||(i.success("Saved .env"),i.success("Saved .mcp.json")),i.blank();;){let u=d.spinner();u.start("Validating Supabase credentials");let o=await M({url:r.supabaseUrl,serviceKey:r.supabaseServiceKey});if(o.ok){u.stop("Supabase credentials accepted");break}u.stop(o.message??"Supabase check failed",2);let p=o.message??"Supabase check failed";i.blank(),i.warn("Bouncing you back into the Supabase section so you can fix it in place."),i.detail("(Press Enter through any field to keep its current value. Ctrl+C to exit.)"),i.blank(),r=await y(r,{onlySection:"supabase",banner:`Supabase rejected your credentials:
@@ -42,6 +42,6 @@ provider to Twilio.`}),await S(n,r)}if(r.telegramBotToken&&r.telegramChatId)for(
42
42
  ${p}
43
43
 
44
44
  Re-enter the bot token or chat_id below.
45
- Press Enter on any field to keep its current value.`}),await S(n,r)}i.blank();let f=d.spinner();f.start("Downloading Klaudius template");let k;try{k=(await O({licenseKey:a,machineId:b(),hostname:process.platform==="win32"?process.env.COMPUTERNAME:void 0,os:`${process.platform}-${process.arch}`,destDir:n})).filesExtracted,f.stop(`Downloaded ${k} files into ${m.bold(n)}`)}catch(u){f.stop(u.message,2),i.detail("Your .env / .mcp.json are saved. Re-run init once the issue is resolved to retry the download."),process.exit(1)}await ye(x(n,"clients"),{recursive:!0});let g=await W(n);if(await L(n,Z("0.1.0",g)),i.success("Wrote .klaudius/manifest.json"),e.skipSchema)i.warn("--skip-schema set; not applying scripts/schema.sql to your Supabase project."),i.detail(`Apply manually later via the SQL editor: ${r.supabaseUrl.replace("https://","https://supabase.com/dashboard/project/"+r.supabaseProjectRef+"/sql/new").replace(r.supabaseProjectRef+".supabase.co","")}`);else{i.blank();let u=d.spinner();u.start("Applying schema to Supabase");let o=await ne({projectRoot:n,pat:r.supabasePat,projectRef:r.supabaseProjectRef});o.ok?u.stop("Schema applied (or already present)"):(u.stop(o.message??"Schema apply failed",2),i.warn("Apply scripts/schema.sql manually in your Supabase SQL editor before running the pipeline."))}if(e.skipInstall)i.warn("--skip-install set; you'll need to run npm install + pip install + playwright install yourself.");else{i.blank(),i.heading("Installing dependencies");let u=await fe(n);i.detail(`npm: ${u.npm}, pip: ${u.pip}, playwright: ${u.playwright}`),(u.npm==="failed"||u.pip==="failed"||u.playwright==="failed")&&i.warn("Some installers failed. You can re-run them manually with `bash setup.sh`.")}i.blank(),d.outro(m.green(m.bold("Project ready."))),console.log(m.dim("Next steps:")),n!==process.cwd()&&console.log(` ${m.cyan(`cd ${e.target}`)}`),console.log(` ${m.cyan("claude")} ${m.dim("# open Claude Code in this project")}`),console.log(` ${m.dim("then ask Claude:")}`),console.log(` ${m.cyan(' "Run the pipeline."')}`),i.blank(),console.log(m.dim("Klaudius ships with sensible defaults for tone, follow-up cadence, pitch")),console.log(m.dim("and more \u2014 but you don't have to live with them. Customise by talking to")),console.log(m.dim("Claude inside this project. Examples you can ask:")),console.log(m.dim(' "Make the outreach more casual."')),console.log(m.dim(' "Change the follow-up cadence to weekly."')),console.log(m.dim(` "Don't pitch businesses with fewer than 5 reviews."`)),console.log(m.dim(' "Use a different opening hook for restaurants."')),console.log(m.dim("Claude knows where to make the change \u2014 no config files to edit by hand.")),i.blank()}async function S(e,s){await ye(e,{recursive:!0}),await ge(x(e,".env"),C(s),"utf-8"),await ge(x(e,".mcp.json"),v(s),"utf-8")}async function nt(e){try{(await tt(e)).isDirectory()||(i.error(`Target exists but is not a directory: ${e}`),process.exit(1))}catch{return{kind:"fresh"}}let s=await et(e),n=s.filter(r=>!r.startsWith(".DS_Store"));if(n.length===0)return{kind:"fresh"};let t=new Set([".env",".mcp.json"]),a=n.every(r=>t.has(r)),l=n.includes(".env"),c=s.includes(".klaudius");return a&&l&&!c?{kind:"partial",envPath:x(e,".env")}:{kind:"occupied",entries:n}}async function rt(e){i.blank(),i.warn(`Found a partial setup at ${m.bold(e)}.`),i.detail(`A previous init wrote ${m.bold(".env")} but didn't finish scaffolding.`),i.blank();let s=await d.select({message:"How would you like to proceed?",initialValue:"resume",options:[{value:"resume",label:"Resume \u2014 use my saved .env values, skip the wizard, retry validation"},{value:"redo",label:"Restart \u2014 walk through the wizard again (existing values pre-filled as defaults)"}]});return d.isCancel(s)&&(d.cancel("Setup cancelled. Existing .env left in place."),process.exit(0)),s}async function at(e){if(e&&e.trim().length>0)return e;d.intro("Klaudius");let s=await d.text({message:"Enter your Klaudius license key",placeholder:"KLAU-XXXX-XXXX-XXXX-XXXX",validate:n=>n.trim().length===0?"Required. Check your purchase confirmation email.":void 0});return d.isCancel(s)&&(d.cancel("Setup cancelled. No files written."),process.exit(0)),s}import{readFile as ot,writeFile as ke,access as lt}from"fs/promises";import{join as D}from"path";import*as $ from"@clack/prompts";import ct from"picocolors";async function Ee(e){let s=e.projectRoot??process.cwd(),n=D(s,".env"),t=D(s,".mcp.json");try{await lt(D(s,".klaudius","manifest.json"))}catch{i.error("This doesn't look like a Klaudius project."),i.detail("Expected `.klaudius/manifest.json`. Run `npx klaudius init` first."),process.exit(1)}let a={};try{let r=await ot(n,"utf-8");a=A(r)}catch{i.warn("No .env found. Starting from an empty configuration.")}let l=z(a);$.intro("Reconfigure Klaudius"),i.detail("Press Enter on any prompt to keep its current value."),i.blank();let c=await y(l);if(c.supabaseUrl!==a.SUPABASE_URL||c.supabaseServiceKey!==a.SUPABASE_SERVICE_KEY){i.heading("Re-validating Supabase credentials");let r=await M({url:c.supabaseUrl,serviceKey:c.supabaseServiceKey});r.ok||(i.error(r.message??"Supabase check failed"),i.warn("Aborting. .env not modified."),process.exit(1)),i.success("Supabase credentials accepted")}await ke(n,C(c),"utf-8"),i.success("Updated .env"),await ke(t,v(c),"utf-8"),i.success("Updated .mcp.json"),i.blank(),$.outro(ct.green("Configuration updated."))}import{copyFile as Se,mkdir as Pe,readdir as ut,readFile as pt,rm as N,writeFile as Ae,access as dt}from"fs/promises";import{dirname as _e,join as w,relative as mt}from"path";import{tmpdir as ft}from"os";import*as T from"@clack/prompts";import H from"picocolors";var j="0.5.0";async function Ie(e){let s=e.projectRoot??process.cwd(),n=await Q(s);n||(i.error("This doesn't look like a Klaudius project."),i.detail("Expected `.klaudius/manifest.json`. Run `npx klaudius init` first."),process.exit(1));let t,a=w(s,".env");try{let o=await pt(a,"utf-8"),p=A(o);t=p.KLAUDIUS_LICENSE_KEY??"",t||(i.error("Couldn't find KLAUDIUS_LICENSE_KEY in your .env file."),i.detail("This file is created by `npx klaudius init`. If your project predates"),i.detail("the licence-gated template (CLI version 0.1.x), re-run `npx klaudius init`"),i.detail("in a sibling directory, then copy KLAUDIUS_LICENSE_KEY from its .env."),process.exit(1));let P=Y(o,p,[{key:"PRICING_TERMS",value:"one-off, no monthly fees",comment:"Phrasing slotted next to ${PRICING} in outreach copy. See template/CLAUDE.md."}]);P!==null&&(await Ae(a,P,"utf-8"),i.detail("Backfilled new env keys: PRICING_TERMS"))}catch{i.error("Couldn't read .env from this directory."),i.detail("Are you running `klaudius update` from inside a Klaudius project?"),process.exit(1)}T.intro(`Update from ${n.templateVersion} \u2192 ${j}`);let l=w(ft(),`klaudius-update-${process.pid}-${Date.now()}`),c;try{i.heading("Downloading the latest Klaudius template");let o=await O({licenseKey:t,machineId:b(),hostname:process.env.HOSTNAME??process.env.COMPUTERNAME,os:`${process.platform}-${process.arch}`,destDir:l});i.success(`Fetched ${o.filesExtracted} canonical files`),c=l}catch(o){i.error(o.message),i.detail("Update aborted. Your project is unchanged."),await N(l,{recursive:!0,force:!0}).catch(()=>{}),process.exit(1)}let r=await ht(c,s,n),h=r.filter(o=>o.state==="new"),f=r.filter(o=>o.state==="clean-update"),k=r.filter(o=>o.state==="user-only"),g=r.filter(o=>o.state==="conflict"),u=r.filter(o=>o.state==="no-change");if(i.blank(),i.heading("Change summary"),i.detail(`${I(u.length,"file")} unchanged`),i.detail(`${I(h.length,"new file")}`),i.detail(`${I(f.length,"file")} updated cleanly (no local edits)`),i.detail(`${I(k.length,"file")} you've edited (no upstream change \u2014 left alone)`),i.detail(`${I(g.length,"file")} with conflicts (you edited AND we have a new version)`),f.length===0&&h.length===0&&g.length===0){i.blank(),T.outro(H.green("Already up to date.")),await N(l,{recursive:!0,force:!0}).catch(()=>{});return}i.blank(),i.heading("Applying clean updates");for(let o of[...h,...f]){let p=w(c,o.path),P=w(s,o.path);await Pe(_e(P),{recursive:!0}),await Se(p,P),n.files[o.path]=o.canonicalHash}if(i.success(`Wrote ${I(h.length+f.length,"file")}`),g.length>0){i.blank(),i.heading(`${g.length} conflict(s) need resolution`);let o=w(s,".klaudius","incoming");try{await N(o,{recursive:!0,force:!0})}catch{}for(let p of g){let P=w(c,p.path),B=w(o,p.path);await Pe(_e(B),{recursive:!0}),await Se(P,B)}await Ae(w(s,".klaudius","conflicts.md"),gt(g,n.templateVersion,j),"utf-8"),i.detail("Staged canonical versions to .klaudius/incoming/"),i.detail("Wrote conflict report to .klaudius/conflicts.md")}n.updatedAt=new Date().toISOString(),g.length===0&&(n.templateVersion=j),await L(s,n),i.blank(),g.length===0?T.outro(H.green("Update complete.")):T.outro(H.yellow(`${g.length} conflict(s) staged. Open Claude Code in this project and ask:
45
+ Press Enter on any field to keep its current value.`}),await S(n,r)}i.blank();let f=d.spinner();f.start("Downloading Klaudius template");let k;try{k=(await O({licenseKey:a,machineId:b(),hostname:process.platform==="win32"?process.env.COMPUTERNAME:void 0,os:`${process.platform}-${process.arch}`,destDir:n})).filesExtracted,f.stop(`Downloaded ${k} files into ${m.bold(n)}`)}catch(u){f.stop(u.message,2),i.detail("Your .env / .mcp.json are saved. Re-run init once the issue is resolved to retry the download."),process.exit(1)}await ye(x(n,"clients"),{recursive:!0});let g=await W(n);if(await L(n,Z("0.1.0",g)),i.success("Wrote .klaudius/manifest.json"),e.skipSchema)i.warn("--skip-schema set; not applying scripts/schema.sql to your Supabase project."),i.detail(`Apply manually later via the SQL editor: ${r.supabaseUrl.replace("https://","https://supabase.com/dashboard/project/"+r.supabaseProjectRef+"/sql/new").replace(r.supabaseProjectRef+".supabase.co","")}`);else{i.blank();let u=d.spinner();u.start("Applying schema to Supabase");let o=await ne({projectRoot:n,pat:r.supabasePat,projectRef:r.supabaseProjectRef});o.ok?u.stop("Schema applied (or already present)"):(u.stop(o.message??"Schema apply failed",2),i.warn("Apply scripts/schema.sql manually in your Supabase SQL editor before running the pipeline."))}if(e.skipInstall)i.warn("--skip-install set; you'll need to run npm install + pip install + playwright install yourself.");else{i.blank(),i.heading("Installing dependencies");let u=await fe(n);i.detail(`npm: ${u.npm}, pip: ${u.pip}, playwright: ${u.playwright}`),(u.npm==="failed"||u.pip==="failed"||u.playwright==="failed")&&i.warn("Some installers failed. You can re-run them manually with `bash setup.sh`.")}i.blank(),d.outro(m.green(m.bold("Project ready."))),console.log(m.dim("Next steps:")),n!==process.cwd()&&console.log(` ${m.cyan(`cd ${e.target}`)}`),console.log(` ${m.cyan("claude")} ${m.dim("# open Claude Code in this project")}`),console.log(` ${m.dim("then ask Claude:")}`),console.log(` ${m.cyan(' "Run the pipeline."')}`),i.blank(),console.log(m.dim("Klaudius ships with sensible defaults for tone, follow-up cadence, pitch")),console.log(m.dim("and more \u2014 but you don't have to live with them. Customise by talking to")),console.log(m.dim("Claude inside this project. Examples you can ask:")),console.log(m.dim(' "Make the outreach more casual."')),console.log(m.dim(' "Change the follow-up cadence to weekly."')),console.log(m.dim(` "Don't pitch businesses with fewer than 5 reviews."`)),console.log(m.dim(' "Use a different opening hook for restaurants."')),console.log(m.dim("Claude knows where to make the change \u2014 no config files to edit by hand.")),i.blank()}async function S(e,s){await ye(e,{recursive:!0}),await ge(x(e,".env"),C(s),"utf-8"),await ge(x(e,".mcp.json"),v(s),"utf-8")}async function nt(e){try{(await tt(e)).isDirectory()||(i.error(`Target exists but is not a directory: ${e}`),process.exit(1))}catch{return{kind:"fresh"}}let s=await et(e),n=s.filter(r=>!r.startsWith(".DS_Store"));if(n.length===0)return{kind:"fresh"};let t=new Set([".env",".mcp.json"]),a=n.every(r=>t.has(r)),l=n.includes(".env"),c=s.includes(".klaudius");return a&&l&&!c?{kind:"partial",envPath:x(e,".env")}:{kind:"occupied",entries:n}}async function rt(e){i.blank(),i.warn(`Found a partial setup at ${m.bold(e)}.`),i.detail(`A previous init wrote ${m.bold(".env")} but didn't finish scaffolding.`),i.blank();let s=await d.select({message:"How would you like to proceed?",initialValue:"resume",options:[{value:"resume",label:"Resume \u2014 use my saved .env values, skip the wizard, retry validation"},{value:"redo",label:"Restart \u2014 walk through the wizard again (existing values pre-filled as defaults)"}]});return d.isCancel(s)&&(d.cancel("Setup cancelled. Existing .env left in place."),process.exit(0)),s}async function at(e){if(e&&e.trim().length>0)return e;d.intro("Klaudius");let s=await d.text({message:"Enter your Klaudius license key",placeholder:"KLAU-XXXX-XXXX-XXXX-XXXX",validate:n=>n.trim().length===0?"Required. Check your purchase confirmation email.":void 0});return d.isCancel(s)&&(d.cancel("Setup cancelled. No files written."),process.exit(0)),s}import{readFile as ot,writeFile as ke,access as lt}from"fs/promises";import{join as D}from"path";import*as $ from"@clack/prompts";import ct from"picocolors";async function Ee(e){let s=e.projectRoot??process.cwd(),n=D(s,".env"),t=D(s,".mcp.json");try{await lt(D(s,".klaudius","manifest.json"))}catch{i.error("This doesn't look like a Klaudius project."),i.detail("Expected `.klaudius/manifest.json`. Run `npx klaudius init` first."),process.exit(1)}let a={};try{let r=await ot(n,"utf-8");a=A(r)}catch{i.warn("No .env found. Starting from an empty configuration.")}let l=z(a);$.intro("Reconfigure Klaudius"),i.detail("Press Enter on any prompt to keep its current value."),i.blank();let c=await y(l);if(c.supabaseUrl!==a.SUPABASE_URL||c.supabaseServiceKey!==a.SUPABASE_SERVICE_KEY){i.heading("Re-validating Supabase credentials");let r=await M({url:c.supabaseUrl,serviceKey:c.supabaseServiceKey});r.ok||(i.error(r.message??"Supabase check failed"),i.warn("Aborting. .env not modified."),process.exit(1)),i.success("Supabase credentials accepted")}await ke(n,C(c),"utf-8"),i.success("Updated .env"),await ke(t,v(c),"utf-8"),i.success("Updated .mcp.json"),i.blank(),$.outro(ct.green("Configuration updated."))}import{copyFile as Se,mkdir as Pe,readdir as ut,readFile as pt,rm as N,writeFile as Ae,access as dt}from"fs/promises";import{dirname as _e,join as w,relative as mt}from"path";import{tmpdir as ft}from"os";import*as T from"@clack/prompts";import H from"picocolors";var j="0.5.1";async function Ie(e){let s=e.projectRoot??process.cwd(),n=await Q(s);n||(i.error("This doesn't look like a Klaudius project."),i.detail("Expected `.klaudius/manifest.json`. Run `npx klaudius init` first."),process.exit(1));let t,a=w(s,".env");try{let o=await pt(a,"utf-8"),p=A(o);t=p.KLAUDIUS_LICENSE_KEY??"",t||(i.error("Couldn't find KLAUDIUS_LICENSE_KEY in your .env file."),i.detail("This file is created by `npx klaudius init`. If your project predates"),i.detail("the licence-gated template (CLI version 0.1.x), re-run `npx klaudius init`"),i.detail("in a sibling directory, then copy KLAUDIUS_LICENSE_KEY from its .env."),process.exit(1));let P=Y(o,p,[{key:"PRICING_TERMS",value:"one-off, no monthly fees",comment:"Phrasing slotted next to ${PRICING} in outreach copy. See template/CLAUDE.md."}]);P!==null&&(await Ae(a,P,"utf-8"),i.detail("Backfilled new env keys: PRICING_TERMS"))}catch{i.error("Couldn't read .env from this directory."),i.detail("Are you running `klaudius update` from inside a Klaudius project?"),process.exit(1)}T.intro(`Update from ${n.templateVersion} \u2192 ${j}`);let l=w(ft(),`klaudius-update-${process.pid}-${Date.now()}`),c;try{i.heading("Downloading the latest Klaudius template");let o=await O({licenseKey:t,machineId:b(),hostname:process.env.HOSTNAME??process.env.COMPUTERNAME,os:`${process.platform}-${process.arch}`,destDir:l});i.success(`Fetched ${o.filesExtracted} canonical files`),c=l}catch(o){i.error(o.message),i.detail("Update aborted. Your project is unchanged."),await N(l,{recursive:!0,force:!0}).catch(()=>{}),process.exit(1)}let r=await ht(c,s,n),h=r.filter(o=>o.state==="new"),f=r.filter(o=>o.state==="clean-update"),k=r.filter(o=>o.state==="user-only"),g=r.filter(o=>o.state==="conflict"),u=r.filter(o=>o.state==="no-change");if(i.blank(),i.heading("Change summary"),i.detail(`${I(u.length,"file")} unchanged`),i.detail(`${I(h.length,"new file")}`),i.detail(`${I(f.length,"file")} updated cleanly (no local edits)`),i.detail(`${I(k.length,"file")} you've edited (no upstream change \u2014 left alone)`),i.detail(`${I(g.length,"file")} with conflicts (you edited AND we have a new version)`),f.length===0&&h.length===0&&g.length===0){i.blank(),T.outro(H.green("Already up to date.")),await N(l,{recursive:!0,force:!0}).catch(()=>{});return}i.blank(),i.heading("Applying clean updates");for(let o of[...h,...f]){let p=w(c,o.path),P=w(s,o.path);await Pe(_e(P),{recursive:!0}),await Se(p,P),n.files[o.path]=o.canonicalHash}if(i.success(`Wrote ${I(h.length+f.length,"file")}`),g.length>0){i.blank(),i.heading(`${g.length} conflict(s) need resolution`);let o=w(s,".klaudius","incoming");try{await N(o,{recursive:!0,force:!0})}catch{}for(let p of g){let P=w(c,p.path),B=w(o,p.path);await Pe(_e(B),{recursive:!0}),await Se(P,B)}await Ae(w(s,".klaudius","conflicts.md"),gt(g,n.templateVersion,j),"utf-8"),i.detail("Staged canonical versions to .klaudius/incoming/"),i.detail("Wrote conflict report to .klaudius/conflicts.md")}n.updatedAt=new Date().toISOString(),g.length===0&&(n.templateVersion=j),await L(s,n),i.blank(),g.length===0?T.outro(H.green("Update complete.")):T.outro(H.yellow(`${g.length} conflict(s) staged. Open Claude Code in this project and ask:
46
46
  "Run /resolve-conflicts"`)),await N(l,{recursive:!0,force:!0}).catch(()=>{})}async function ht(e,s,n){let t=await Te(e),a=[];for(let l of t){let c=await U(w(e,l)),r=n.files[l],h=w(s,l),f;try{await dt(h),f=await U(h)}catch{a.push({path:l,state:"new",canonicalHash:c,manifestHash:r});continue}c===r?f===r?a.push({path:l,state:"no-change",canonicalHash:c,localHash:f,manifestHash:r}):a.push({path:l,state:"user-only",canonicalHash:c,localHash:f,manifestHash:r}):f===r?a.push({path:l,state:"clean-update",canonicalHash:c,localHash:f,manifestHash:r}):a.push({path:l,state:"conflict",canonicalHash:c,localHash:f,manifestHash:r})}return a}function I(e,s){return`${e} ${s}${e===1?"":"s"}`}async function Te(e,s=e){let n=await ut(e,{withFileTypes:!0}),t=[];for(let a of n){let l=w(e,a.name);a.isDirectory()?t.push(...await Te(l,s)):a.isFile()&&t.push(mt(s,l))}return t}function gt(e,s,n){let t=[];t.push("# Update Conflicts"),t.push(""),t.push(`Generated by \`npx klaudius update\` on ${new Date().toISOString()}.`),t.push(`Updating from canonical version **${s}** to **${n}**.`),t.push(""),t.push(`${e.length} file${e.length===1?"":"s"} ha${e.length===1?"s":"ve"} BOTH local edits AND upstream changes.`),t.push("Open Claude Code in this project and run `/resolve-conflicts` \u2014 that skill walks through each conflict interactively, helping you decide what to keep, take, or merge."),t.push(""),t.push("---"),t.push("");for(let a of e)t.push(`## ${a.path}`),t.push(""),t.push(`- Your version (with your local edits): \`${a.path}\``),t.push(`- New canonical version: \`.klaudius/incoming/${a.path}\``),t.push(`- Hash at last install/update: \`${(a.manifestHash??"(none)").slice(0,16)}\``),t.push(`- Your local hash now: \`${(a.localHash??"(missing)").slice(0,16)}\``),t.push(`- New canonical hash: \`${a.canonicalHash.slice(0,16)}\``),t.push(""),t.push("**Resolution options for the SKILL to walk through:**"),t.push(""),t.push("1. Keep mine \u2014 discard the new canonical changes"),t.push("2. Take canonical \u2014 overwrite my edits with the new version"),t.push("3. Merge \u2014 review both side-by-side and produce a combined version"),t.push(""),t.push("---"),t.push("");return t.join(`
47
- `)}var wt="0.5.0",R=new yt;R.name("klaudius").description("Scaffold and maintain a Klaudius pipeline project").version(wt);R.command("init").description("Scaffold a new Klaudius project").argument("[target]","Directory to create (defaults to current directory)",".").option("--license <key>","License key (received in your purchase confirmation email)").option("--skip-install","Skip npm/pip/playwright installation").option("--skip-schema","Skip applying schema.sql to your Supabase project").action(async(e,s)=>{try{await we({target:e,license:s.license,skipInstall:!!s.skipInstall,skipSchema:!!s.skipSchema})}catch(n){console.error("init failed:",n.message),process.exit(1)}});R.command("configure").description("Re-run the setup wizard with current values pre-filled").action(async()=>{try{await Ee({})}catch(e){console.error("configure failed:",e.message),process.exit(1)}});R.command("update").description("Pull latest template; produce a conflicts report for any files you've modified").action(async()=>{try{await Ie({})}catch(e){console.error("update failed:",e.message),process.exit(1)}});R.parse(process.argv);
47
+ `)}var wt="0.5.1",R=new yt;R.name("klaudius").description("Scaffold and maintain a Klaudius pipeline project").version(wt);R.command("init").description("Scaffold a new Klaudius project").argument("[target]","Directory to create (defaults to current directory)",".").option("--license <key>","License key (received in your purchase confirmation email)").option("--skip-install","Skip npm/pip/playwright installation").option("--skip-schema","Skip applying schema.sql to your Supabase project").action(async(e,s)=>{try{await we({target:e,license:s.license,skipInstall:!!s.skipInstall,skipSchema:!!s.skipSchema})}catch(n){console.error("init failed:",n.message),process.exit(1)}});R.command("configure").description("Re-run the setup wizard with current values pre-filled").action(async()=>{try{await Ee({})}catch(e){console.error("configure failed:",e.message),process.exit(1)}});R.command("update").description("Pull latest template; produce a conflicts report for any files you've modified").action(async()=>{try{await Ie({})}catch(e){console.error("update failed:",e.message),process.exit(1)}});R.parse(process.argv);
package/package.json CHANGED
@@ -1,12 +1,12 @@
1
1
  {
2
2
  "name": "klaudius",
3
- "version": "0.5.0",
3
+ "version": "0.5.1",
4
4
  "description": "Scaffold and maintain a Klaudius project — an autonomous web agency pipeline that runs on Claude Code (find, build, deploy, outreach end-to-end).",
5
5
  "homepage": "https://klaudius.dev",
6
6
  "bugs": {
7
7
  "email": "hello@klaudius.dev"
8
8
  },
9
- "author": "Cloudbot Limited",
9
+ "author": "Klaudius",
10
10
  "license": "UNLICENSED",
11
11
  "type": "module",
12
12
  "bin": {