klaudius 0.5.1 → 0.7.0

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
@@ -24,6 +24,7 @@ The `init` wizard prompts for credentials (Supabase, Vercel, your email and SMS
24
24
  npx klaudius init [target] # Scaffold a fresh project
25
25
  npx klaudius configure # Re-run the setup wizard against an existing project
26
26
  npx klaudius update # Pull latest template; conflicts staged for resolution
27
+ npx klaudius doctor # Diagnose a broken install (env, files, deps, credentials)
27
28
  ```
28
29
 
29
30
  ## What gets scaffolded
@@ -1,4 +1,4 @@
1
- import{a as g,d as Q}from"./chunk-GI7NTKNT.js";import{useEffect as $e,useMemo as je,useRef as Z,useState as T}from"react";import{Box as P,Text as I,useApp as We,useInput as qe}from"ink";var ee={gmail:{label:"Gmail",smtpHost:"smtp.gmail.com",smtpPort:587,imapHost:"imap.gmail.com",imapPort:993,sentFolder:"[Gmail]/Sent Mail",inboxFolder:"INBOX",appPasswordHelp:`Generate at https://myaccount.google.com/apppasswords
1
+ import{a as f,e as Q}from"./chunk-SZ4KCTSL.js";import{useEffect as We,useMemo as je,useRef as Z,useState as T}from"react";import{Box as P,Text as k,useApp as qe,useInput as Ye}from"ink";var ee={gmail:{label:"Gmail",smtpHost:"smtp.gmail.com",smtpPort:587,imapHost:"imap.gmail.com",imapPort:993,sentFolder:"[Gmail]/Sent Mail",inboxFolder:"INBOX",appPasswordHelp:`Generate at https://myaccount.google.com/apppasswords
2
2
  (2-step verification must be enabled on your Google account first;
3
3
  Google won't accept your normal account password from third-party apps).`},outlook:{label:"Outlook / Office 365",smtpHost:"smtp-mail.outlook.com",smtpPort:587,imapHost:"outlook.office365.com",imapPort:993,sentFolder:"Sent Items",inboxFolder:"INBOX",appPasswordHelp:`Generate at https://account.microsoft.com/security \u2192 Advanced security options \u2192 App passwords
4
4
  (2-step verification must be enabled first).`},ionos:{label:"IONOS",smtpHost:"smtp.ionos.co.uk",smtpPort:587,imapHost:"imap.ionos.co.uk",imapPort:993,sentFolder:"Sent Items",inboxFolder:"INBOX",appPasswordHelp:`Use your normal IONOS mailbox password.
@@ -6,31 +6,31 @@ Google won't accept your normal account password from third-party apps).`},outlo
6
6
  (Fastmail requires app passwords for IMAP/SMTP regardless of 2FA setting.)`},zoho:{label:"Zoho Mail",smtpHost:"smtp.zoho.com",smtpPort:587,imapHost:"imap.zoho.com",imapPort:993,sentFolder:"Sent",inboxFolder:"INBOX",appPasswordHelp:`Generate at https://accounts.zoho.com/u/h#security/app_passwords
7
7
  (Zoho requires app passwords for IMAP/SMTP \u2014 your normal account password won't work.)`}},te=`Use the app-specific password your provider issues for SMTP/IMAP
8
8
  (check their security settings \u2014 most providers require a generated password
9
- rather than your normal account password).`;var N=class extends Error{constructor(){super("user pressed back"),this.name="BackSignal"}},U=class extends Error{constructor(){super("user cancelled"),this.name="CancelSignal"}},Se="Include the country code with a + (e.g. +12025550123 for US, +447123456789 for UK)",oe=e=>/^\+\d{8,15}$/.test(e)?void 0:Se,Pe=e=>/^[^@\s]+@[^@\s]+\.[^@\s]+$/.test(e)?void 0:"Enter a valid email",A=e=>t=>t.trim().length===0?`${e} is required`:void 0,xe="one-off, no monthly fees",ae="per month";function H(e){if(!e)return!0;let t=e.trim();if(t==="")return!0;let n=t.replace(/[^\d.]/g,"");if(n==="")return/^\s*free\s*$/i.test(t);let o=parseFloat(n);return!isNaN(o)&&o===0}var ve={id:"pricing",title:"Pricing",isApplicable:()=>!0,execute:async(e,{prompt:t})=>{let n=e.pricingTerms===ae,o=n?"":e.pricing??"$999",s=e.pricingMonthly??(n?e.pricing??"":""),a=await t({type:"text",message:"What's the upfront fee you'll charge clients per site?",placeholder:"e.g. $999, \xA3799, \u20AC899 \u2014 or 0 for monthly-only billing",initialValue:o,note:{title:"Upfront fee",body:`The one-time amount charged when a client signs up. Type 0 (or
9
+ rather than your normal account password).`;var R=class extends Error{constructor(){super("user pressed back"),this.name="BackSignal"}},U=class extends Error{constructor(){super("user cancelled"),this.name="CancelSignal"}},Pe="Include the country code with a + (e.g. +12025550123 for US, +447123456789 for UK)",oe=e=>/^\+\d{8,15}$/.test(e)?void 0:Pe,ve=e=>/^[^@\s]+@[^@\s]+\.[^@\s]+$/.test(e)?void 0:"Enter a valid email",C=e=>t=>t.trim().length===0?`${e} is required`:void 0,xe="one-off, no monthly fees",ne="per month";function H(e){if(!e)return!0;let t=e.trim();if(t==="")return!0;let n=t.replace(/[^\d.]/g,"");if(n==="")return/^\s*free\s*$/i.test(t);let o=parseFloat(n);return!isNaN(o)&&o===0}var ke={id:"pricing",title:"Pricing",isApplicable:()=>!0,execute:async(e,{prompt:t})=>{let n=e.pricingTerms===ne,o=n?"":e.pricing??"$999",r=e.pricingMonthly??(n?e.pricing??"":""),a=await t({type:"text",message:"What's the upfront fee you'll charge clients per site?",placeholder:"e.g. $999, \xA3799, \u20AC899 \u2014 or 0 for monthly-only billing",initialValue:o,note:{title:"Upfront fee",body:`The one-time amount charged when a client signs up. Type 0 (or
10
10
  leave blank) if you only want to charge a monthly fee. You can
11
11
  also add a recurring monthly fee on the next prompt \u2014 combining
12
12
  the two is fine (e.g. $499 upfront + $49/month).
13
13
 
14
14
  Use the currency that matches your clients' region. Klaudius
15
- uses this string verbatim throughout outreach.`}}),i=await t({type:"text",message:"What's the monthly fee? (optional \u2014 leave blank for one-off only)",placeholder:"e.g. $49, \xA339, \u20AC45 \u2014 or blank for none",initialValue:s,validate:v=>{if(H(a)&&H(v))return"Set an upfront fee, a monthly fee, or both \u2014 they can't both be zero."},note:{title:"Monthly fee (optional)",body:`Leave blank to charge a one-off fee only. Fill it in to charge
15
+ uses this string verbatim throughout outreach.`}}),s=await t({type:"text",message:"What's the monthly fee? (optional \u2014 leave blank for one-off only)",placeholder:"e.g. $49, \xA339, \u20AC45 \u2014 or blank for none",initialValue:r,validate:x=>{if(H(a)&&H(x))return"Set an upfront fee, a monthly fee, or both \u2014 they can't both be zero."},note:{title:"Monthly fee (optional)",body:`Leave blank to charge a one-off fee only. Fill it in to charge
16
16
  a recurring monthly fee \u2014 either on its own (if upfront was 0)
17
- or alongside the upfront fee.`}}),u=H(a),r=H(i),l,p,m;return u&&!r?(l=i.trim(),p=ae,m=i.trim()):!u&&r?(l=a.trim(),p=xe,m=void 0):(l=a.trim(),p=`upfront, then ${i.trim()}/month`,m=i.trim()),{pricing:l,pricingTerms:p,pricingMonthly:m}}},Ie={id:"outreach-enabled",title:"Automated outreach",isApplicable:()=>!0,execute:async(e,{prompt:t})=>({outreachEnabled:await t({type:"confirm",message:"Use Klaudius's automated outreach features?",initialValue:e.outreachEnabled??!0,note:{title:"Automated outreach features",body:`After finding candidates and building websites for them, Klaudius can
17
+ or alongside the upfront fee.`}}),u=H(a),i=H(s),l,c,m;return u&&!i?(l=s.trim(),c=ne,m=s.trim()):!u&&i?(l=a.trim(),c=xe,m=void 0):(l=a.trim(),c=`upfront, then ${s.trim()}/month`,m=s.trim()),{pricing:l,pricingTerms:c,pricingMonthly:m}}},Ie={id:"outreach-enabled",title:"Automated outreach",isApplicable:()=>!0,execute:async(e,{prompt:t})=>({outreachEnabled:await t({type:"confirm",message:"Use Klaudius's automated outreach features?",initialValue:e.outreachEnabled??!0,note:{title:"Automated outreach features",body:`After finding candidates and building websites for them, Klaudius can
18
18
  run outreach to them automatically (email and/or SMS).
19
- If you'd rather handle outreach manually, you can skip this here.`}})})},Te={id:"operator-name",title:"Your name",isApplicable:e=>e.outreachEnabled===!0,execute:async(e,{prompt:t})=>{let n=await t({type:"text",message:"Your name as it should appear publicly in outreach",placeholder:"e.g. Mike Smith \u2014 or your business name if you'd rather",initialValue:e.name??e.emailFromName??"",validate:A("Name")}),o=n.trim().split(/\s+/)[0];return{name:n,signature:e.signature??`Thanks, ${o}`}}},ke={id:"channels",title:"Outreach channels",isApplicable:e=>e.outreachEnabled===!0,execute:async(e,{prompt:t})=>{let n=e.outreachChannels&&e.outreachChannels.length>1?"both":e.outreachChannels?.[0]??"email",o=await t({type:"select",message:"Which channels can Klaudius use for outreach?",initialValue:n,options:[{value:"email",label:"Email only"},{value:"sms",label:"SMS only"},{value:"both",label:"Both (recommended for maximum reach)"}],note:{title:"Outreach channels",body:`Different businesses have different contact methods publicly available \u2014
19
+ If you'd rather handle outreach manually, you can skip this here.`}})})},Te={id:"operator-name",title:"Your name",isApplicable:e=>e.outreachEnabled===!0,execute:async(e,{prompt:t})=>{let n=await t({type:"text",message:"Your name as it should appear publicly in outreach",placeholder:"e.g. Mike Smith \u2014 or your business name if you'd rather",initialValue:e.name??e.emailFromName??"",validate:C("Name")}),o=n.trim().split(/\s+/)[0];return{name:n,signature:e.signature??`Thanks, ${o}`}}},Ae={id:"channels",title:"Outreach channels",isApplicable:e=>e.outreachEnabled===!0,execute:async(e,{prompt:t})=>{let n=e.outreachChannels&&e.outreachChannels.length>1?"both":e.outreachChannels?.[0]??"email",o=await t({type:"select",message:"Which channels can Klaudius use for outreach?",initialValue:n,options:[{value:"email",label:"Email only"},{value:"sms",label:"SMS only"},{value:"both",label:"Both (recommended for maximum reach)"}],note:{title:"Outreach channels",body:`Different businesses have different contact methods publicly available \u2014
20
20
  some list only an email, others only a phone number. Klaudius uses whichever
21
21
  channel is available for each candidate (it doesn't message the same candidate
22
22
  on both). Picking 'both' just means Klaudius can reach more of the
23
- businesses it finds.`}}),s=o==="both"?["email","sms"]:[o],a;return s.length>1&&(a=await t({type:"select",message:"When a candidate has both email and a phone number, which channel should Klaudius try first?",initialValue:e.outreachPriority??"email",options:[{value:"email",label:"Prefer email (better for B2B / professional services)"},{value:"sms",label:"Prefer SMS (better response rates for trades / local services)"}]})),{outreachChannels:s,outreachPriority:a}}},Ae={id:"email",title:"Email setup",isApplicable:e=>e.outreachChannels?.includes("email")===!0,execute:async(e,{prompt:t,info:n})=>{n({title:"Email setup",body:`Klaudius needs an email account to send outreach FROM. This is the
23
+ businesses it finds.`}}),r=o==="both"?["email","sms"]:[o],a;return r.length>1&&(a=await t({type:"select",message:"When a candidate has both email and a phone number, which channel should Klaudius try first?",initialValue:e.outreachPriority??"email",options:[{value:"email",label:"Prefer email (better for B2B / professional services)"},{value:"sms",label:"Prefer SMS (better response rates for trades / local services)"}]})),{outreachChannels:r,outreachPriority:a}}},Ce={id:"email",title:"Email setup",isApplicable:e=>e.outreachChannels?.includes("email")===!0,execute:async(e,{prompt:t,info:n})=>{n({title:"Email setup",body:`Klaudius needs an email account to send outreach FROM. This is the
24
24
  address candidates will see as the sender. Use a domain you control
25
- (your own, your business's) \u2014 not your personal address.`});let o=await t({type:"select",message:"Who hosts the email account you'll send outreach from?",initialValue:e.emailProvider??"gmail",options:[{value:"gmail",label:"Gmail"},{value:"fastmail",label:"Fastmail"},{value:"outlook",label:"Outlook / Office 365"},{value:"zoho",label:"Zoho Mail"},{value:"ionos",label:"IONOS"},{value:"custom",label:"Custom / other (enter SMTP/IMAP details manually)"}]}),s=o!=="custom"?ee[o]:void 0,a=await t({type:"text",message:"Email address to send outreach from",placeholder:"you@yourbusiness.com",initialValue:e.emailAddress??"",validate:Pe}),i=s?.appPasswordHelp??te,u=await t({type:"password",message:"Paste the app password",initialValue:e.emailPassword??"",validate:A("App password"),note:{title:"App password",body:`An app password is a separate, single-purpose password that your
25
+ (your own, your business's) \u2014 not your personal address.`});let o=await t({type:"select",message:"Who hosts the email account you'll send outreach from?",initialValue:e.emailProvider??"gmail",options:[{value:"gmail",label:"Gmail"},{value:"fastmail",label:"Fastmail"},{value:"outlook",label:"Outlook / Office 365"},{value:"zoho",label:"Zoho Mail"},{value:"ionos",label:"IONOS"},{value:"custom",label:"Custom / other (enter SMTP/IMAP details manually)"}]}),r=o!=="custom"?ee[o]:void 0,a=await t({type:"text",message:"Email address to send outreach from",placeholder:"you@yourbusiness.com",initialValue:e.emailAddress??"",validate:ve}),s=r?.appPasswordHelp??te,u=await t({type:"password",message:"Paste the app password",initialValue:e.emailPassword??"",validate:C("App password"),note:{title:"App password",body:`An app password is a separate, single-purpose password that your
26
26
  email provider issues for third-party apps like Klaudius. It is NOT
27
27
  your normal account login password.
28
28
 
29
- `+i+`
29
+ `+s+`
30
30
 
31
31
  Once generated, it's stored locally in this project's .env file.
32
32
  It is never sent to Klaudius's servers \u2014 only to your email
33
- provider when sending outreach.`}}),r,l,p,m,v,E;if(o==="custom"){r=await t({type:"text",message:"SMTP host (the server Klaudius connects to for sending)",placeholder:"e.g. smtp.yourprovider.com",initialValue:e.emailSmtpHost??"",validate:A("SMTP host")});let F=await t({type:"text",message:"SMTP port (usually 587 for STARTTLS or 465 for SSL)",initialValue:String(e.emailSmtpPort??587),validate:k=>/^\d+$/.test(k)?void 0:"Must be a number"});l=parseInt(F,10),p=await t({type:"text",message:"IMAP host (the server Klaudius connects to for reading replies)",placeholder:"e.g. imap.yourprovider.com",initialValue:e.emailImapHost??"",validate:A("IMAP host")});let O=await t({type:"text",message:"IMAP port (usually 993 for SSL)",initialValue:String(e.emailImapPort??993),validate:k=>/^\d+$/.test(k)?void 0:"Must be a number"});m=parseInt(O,10),v=await t({type:"text",message:"Sent folder name (so SMTP-sent messages also appear in your email client's Sent folder)",placeholder:'e.g. "Sent", "Sent Items"',initialValue:e.emailSentFolder??"Sent"}),E=await t({type:"text",message:"Inbox folder name (almost always INBOX)",initialValue:e.emailInboxFolder??"INBOX"})}else r=s.smtpHost,l=s.smtpPort,p=s.imapHost,m=s.imapPort,v=s.sentFolder,E=s.inboxFolder;return{emailProvider:o,emailAddress:a,emailPassword:u,emailFromName:e.name,emailSmtpHost:r,emailSmtpPort:l,emailImapHost:p,emailImapPort:m,emailSentFolder:v,emailInboxFolder:E}}},Ce={id:"sms",title:"SMS setup",isApplicable:e=>e.outreachChannels?.includes("sms")===!0,execute:async(e,{prompt:t,info:n})=>{n({title:"SMS setup",body:`Klaudius can send SMS one of two ways:
33
+ provider when sending outreach.`}}),i,l,c,m,x,E;if(o==="custom"){i=await t({type:"text",message:"SMTP host (the server Klaudius connects to for sending)",placeholder:"e.g. smtp.yourprovider.com",initialValue:e.emailSmtpHost??"",validate:C("SMTP host")});let _=await t({type:"text",message:"SMTP port (usually 587 for STARTTLS or 465 for SSL)",initialValue:String(e.emailSmtpPort??587),validate:A=>/^\d+$/.test(A)?void 0:"Must be a number"});l=parseInt(_,10),c=await t({type:"text",message:"IMAP host (the server Klaudius connects to for reading replies)",placeholder:"e.g. imap.yourprovider.com",initialValue:e.emailImapHost??"",validate:C("IMAP host")});let O=await t({type:"text",message:"IMAP port (usually 993 for SSL)",initialValue:String(e.emailImapPort??993),validate:A=>/^\d+$/.test(A)?void 0:"Must be a number"});m=parseInt(O,10),x=await t({type:"text",message:"Sent folder name (so SMTP-sent messages also appear in your email client's Sent folder)",placeholder:'e.g. "Sent", "Sent Items"',initialValue:e.emailSentFolder??"Sent"}),E=await t({type:"text",message:"Inbox folder name (almost always INBOX)",initialValue:e.emailInboxFolder??"INBOX"})}else i=r.smtpHost,l=r.smtpPort,c=r.imapHost,m=r.imapPort,x=r.sentFolder,E=r.inboxFolder;return{emailProvider:o,emailAddress:a,emailPassword:u,emailFromName:e.name,emailSmtpHost:i,emailSmtpPort:l,emailImapHost:c,emailImapPort:m,emailSentFolder:x,emailInboxFolder:E}}},Ee={id:"sms",title:"SMS setup",isApplicable:e=>e.outreachChannels?.includes("sms")===!0,execute:async(e,{prompt:t,info:n})=>{n({title:"SMS setup",body:`Klaudius can send SMS one of two ways:
34
34
 
35
35
  \u2022 macOS iMessage \u2014 free, uses Messages.app on your Mac.
36
36
  Reaches other Apple users as iMessage for free. To reach
@@ -46,7 +46,7 @@ provider when sending outreach.`}}),r,l,p,m,v,E;if(o==="custom"){r=await t({type
46
46
 
47
47
  \u2022 Twilio \u2014 cross-platform, ~$0.01 per SMS. Sender is a phone
48
48
  number you buy from Twilio. Works on any OS, reaches any
49
- phone, no iPhone required. Recommended for consistent delivery.`});let o=await t({type:"select",message:"Which SMS provider?",initialValue:e.smsProvider??(process.platform==="darwin"?"imessage":"twilio"),options:[{value:"imessage",label:"macOS iMessage (free, Mac only)"},{value:"twilio",label:"Twilio (cross-platform, paid per-message \u2014 recommended)"}]}),s=await t({type:"text",message:"Your mobile number (used as a safe target for test sends so Klaudius doesn't accidentally message real candidates while you're testing)",placeholder:"+12025550123",initialValue:e.testPhone??"",validate:oe}),a={smsProvider:o,testPhone:s};return o==="twilio"&&(a.twilioAccountSid=await t({type:"text",message:"Twilio Account SID (starts with AC...)",initialValue:e.twilioAccountSid??"",validate:i=>/^AC[a-zA-Z0-9]+$/.test(i)?void 0:"Should start with 'AC'"}),a.twilioAuthToken=await t({type:"password",message:"Twilio Auth Token",initialValue:e.twilioAuthToken??"",validate:A("Twilio Auth Token")}),a.twilioPhoneNumber=await t({type:"text",message:"Twilio phone number (the number outreach will be sent FROM)",placeholder:"+15551234567",initialValue:e.twilioPhoneNumber??"",validate:oe})),a}},Ee={id:"vercel",title:"Vercel deployment",isApplicable:()=>!0,execute:async(e,{prompt:t})=>({vercelToken:await t({type:"password",message:"Paste your Vercel API token",initialValue:e.vercelToken??"",validate:o=>o.trim().length===0?"Required \u2014 without this Klaudius can't deploy generated sites":void 0,note:{title:"Vercel deployment",body:`Klaudius deploys every generated client site to Vercel. We use a
49
+ phone, no iPhone required. Recommended for consistent delivery.`});let o=await t({type:"select",message:"Which SMS provider?",initialValue:e.smsProvider??(process.platform==="darwin"?"imessage":"twilio"),options:[{value:"imessage",label:"macOS iMessage (free, Mac only)"},{value:"twilio",label:"Twilio (cross-platform, paid per-message \u2014 recommended)"}]}),r=await t({type:"text",message:"Your mobile number (used as a safe target for test sends so Klaudius doesn't accidentally message real candidates while you're testing)",placeholder:"+12025550123",initialValue:e.testPhone??"",validate:oe}),a={smsProvider:o,testPhone:r};return o==="twilio"&&(a.twilioAccountSid=await t({type:"text",message:"Twilio Account SID (starts with AC...)",initialValue:e.twilioAccountSid??"",validate:s=>/^AC[a-zA-Z0-9]+$/.test(s)?void 0:"Should start with 'AC'"}),a.twilioAuthToken=await t({type:"password",message:"Twilio Auth Token",initialValue:e.twilioAuthToken??"",validate:C("Twilio Auth Token")}),a.twilioPhoneNumber=await t({type:"text",message:"Twilio phone number (the number outreach will be sent FROM)",placeholder:"+15551234567",initialValue:e.twilioPhoneNumber??"",validate:oe})),a}},Oe={id:"vercel",title:"Vercel deployment",isApplicable:()=>!0,execute:async(e,{prompt:t})=>({vercelToken:await t({type:"password",message:"Paste your Vercel API token",initialValue:e.vercelToken??"",validate:o=>o.trim().length===0?"Required \u2014 without this Klaudius can't deploy generated sites":void 0,note:{title:"Vercel deployment",body:`Klaudius deploys every generated client site to Vercel. We use a
50
50
  Vercel API token to do this without an interactive login.
51
51
 
52
52
  How to get one (one-time, ~30 seconds):
@@ -69,7 +69,7 @@ STUCK? Open Claude (or any LLM) in another tab/app and paste:
69
69
  "I need to create a personal Vercel API token for the
70
70
  Klaudius CLI tool. Walk me through generating one at
71
71
  vercel.com/account/tokens, including which scope and
72
- expiration to choose."`}})})},Oe={id:"supabase",title:"Supabase",isApplicable:()=>!0,execute:async(e,{prompt:t,info:n})=>{n({title:"Supabase setup",body:`Supabase is your CRM. It's where Klaudius records every business it
72
+ expiration to choose."`}})})},Me={id:"supabase",title:"Supabase",isApplicable:()=>!0,execute:async(e,{prompt:t,info:n})=>{n({title:"Supabase setup",body:`Supabase is your CRM. It's where Klaudius records every business it
73
73
  finds, where each one sits in your funnel (gathered \u2192 built \u2192 deployed
74
74
  \u2192 outreach sent \u2192 responded), and the follow-up state for every lead.
75
75
  You can browse it at any time from the Supabase dashboard, or ask
@@ -103,11 +103,11 @@ in another tab/app and paste this prompt verbatim:
103
103
  The wizard is asking for: (1) project URL, (2) the secret/service-
104
104
  role key, (3) a personal access token. Walk me through where to
105
105
  find each one in the current Supabase dashboard UI, including the
106
- exact menu paths and any caveats for newer vs. legacy projects."`});let s=(await t({type:"text",message:"Paste your Supabase API URL",placeholder:"https://YOUR_REF.supabase.co",initialValue:e.supabaseUrl??"",validate:r=>/^https:\/\/[a-z0-9]+\.supabase\.co(\/rest\/v1\/?)?$/i.test(r.trim())?void 0:"Should look like https://xxxx.supabase.co (with or without /rest/v1/)",note:{title:"API URL",body:`In your Supabase dashboard, open your project, then go to:
106
+ exact menu paths and any caveats for newer vs. legacy projects."`});let r=(await t({type:"text",message:"Paste your Supabase API URL",placeholder:"https://YOUR_REF.supabase.co",initialValue:e.supabaseUrl??"",validate:i=>/^https:\/\/[a-z0-9]+\.supabase\.co(\/rest\/v1\/?)?$/i.test(i.trim())?void 0:"Should look like https://xxxx.supabase.co (with or without /rest/v1/)",note:{title:"API URL",body:`In your Supabase dashboard, open your project, then go to:
107
107
  Settings \u2192 Data API \u2192 'API URL'
108
108
 
109
109
  It looks like https://xxxx.supabase.co. Click to copy and paste
110
- below.`}})).trim().replace(/\/rest\/v1\/?$/i,"").replace(/\/$/,""),a=s.match(/^https:\/\/([a-z0-9]+)\.supabase\.co/i)?.[1]??"",i=await t({type:"password",message:"Paste the Supabase secret key (sb_secret_\u2026)",initialValue:e.supabaseServiceKey??"",validate:r=>r.length<20?"Looks too short":void 0,note:{title:"Secret key",body:a?`Open this exact page:
110
+ below.`}})).trim().replace(/\/rest\/v1\/?$/i,"").replace(/\/$/,""),a=r.match(/^https:\/\/([a-z0-9]+)\.supabase\.co/i)?.[1]??"",s=await t({type:"password",message:"Paste the Supabase secret key (sb_secret_\u2026)",initialValue:e.supabaseServiceKey??"",validate:i=>i.length<20?"Looks too short":void 0,note:{title:"Secret key",body:a?`Open this exact page:
111
111
  https://supabase.com/dashboard/project/${a}/settings/api-keys
112
112
 
113
113
  Under 'Secret keys', find the row whose value starts with
@@ -115,7 +115,7 @@ Under 'Secret keys', find the row whose value starts with
115
115
 
116
116
  NEVER use the 'anon' / 'public' / 'sb_publishable_' key (those
117
117
  are client-side keys with no write permissions).`:`Open Settings \u2192 API Keys in your Supabase project. Copy the
118
- secret key (starts with 'sb_secret_').`}}),u=await t({type:"password",message:"Paste the Supabase personal access token",initialValue:e.supabasePat??"",validate:r=>r.startsWith("sbp_")?void 0:"Should start with 'sbp_'",note:{title:"Personal access token",body:`This is a separate credential from the secret key above \u2014 it's an
118
+ secret key (starts with 'sb_secret_').`}}),u=await t({type:"password",message:"Paste the Supabase personal access token",initialValue:e.supabasePat??"",validate:i=>i.startsWith("sbp_")?void 0:"Should start with 'sbp_'",note:{title:"Personal access token",body:`This is a separate credential from the secret key above \u2014 it's an
119
119
  ACCOUNT-wide token (not project-scoped) used by the Supabase MCP and
120
120
  by Klaudius's schema migration.
121
121
 
@@ -125,7 +125,7 @@ Open this exact page:
125
125
  Click 'Generate new token', name it 'Klaudius', and set the
126
126
  expiration to 'Never' (recommended \u2014 saves you having to rotate
127
127
  it later). Copy the value (starts with 'sbp_') and paste it
128
- below.`}});return{supabaseUrl:s,supabaseServiceKey:i,supabasePat:u,supabaseProjectRef:a}}},Me={id:"google-places",title:"Google Places API",isApplicable:()=>!0,execute:async(e,{prompt:t})=>({googlePlacesApiKey:await t({type:"password",message:"Paste your Google Places API key",placeholder:"AIzaSy\u2026",initialValue:e.googlePlacesApiKey??"",validate:o=>o.trim().length===0?"Required \u2014 without this Klaudius can't find candidates":void 0,note:{title:"Google Places API key",body:`Klaudius's /find skill queries Google's Places API to discover businesses
128
+ below.`}});return{supabaseUrl:r,supabaseServiceKey:s,supabasePat:u,supabaseProjectRef:a}}},Ne={id:"google-places",title:"Google Places API",isApplicable:()=>!0,execute:async(e,{prompt:t})=>({googlePlacesApiKey:await t({type:"password",message:"Paste your Google Places API key",placeholder:"AIzaSy\u2026",initialValue:e.googlePlacesApiKey??"",validate:o=>o.trim().length===0?"Required \u2014 without this Klaudius can't find candidates":void 0,note:{title:"Google Places API key",body:`Klaudius's /find skill queries Google's Places API to discover businesses
129
129
  without websites in your target region. Without a key, the pipeline has
130
130
  no source of leads \u2014 so this is required.
131
131
 
@@ -161,13 +161,13 @@ STUCK? Open Claude (or any LLM) in another tab/app and paste this prompt:
161
161
  "I need a Google Places API (New) key for a CLI tool called
162
162
  Klaudius. Walk me through claiming the $300 Google Cloud free trial
163
163
  and getting the API key Google auto-generates after enabling
164
- Places API (New), including any signup gotchas."`}})})},Ne={id:"telegram",title:"Telegram (optional)",isApplicable:()=>!0,execute:async(e,{prompt:t})=>{if(!await t({type:"confirm",message:"Set up Telegram notifications? (sends a ping when outreach lands)",initialValue:!!e.telegramBotToken,note:{title:"Telegram notifications (optional)",body:`Klaudius can ping a Telegram chat when outreach is sent or when a
164
+ Places API (New), including any signup gotchas."`}})})},Re={id:"telegram",title:"Telegram (optional)",isApplicable:()=>!0,clearOnSkip:["telegramBotToken","telegramChatId"],execute:async(e,{prompt:t})=>{if(!await t({type:"confirm",message:"Set up Telegram notifications? (sends a ping when outreach lands)",initialValue:!!e.telegramBotToken,note:{title:"Telegram notifications (optional)",body:`Klaudius can ping a Telegram chat when outreach is sent or when a
165
165
  candidate replies. This is OPTIONAL \u2014 skip it if you'd rather check
166
166
  the Supabase dashboard manually.
167
167
 
168
168
  If you say yes, you'll need a Telegram bot token (created via
169
169
  @BotFather, ~30 seconds) and your chat ID (your own user ID, for
170
- personal pings). The next two prompts walk you through both.`}}))return{};let o=await t({type:"password",message:"Paste your Telegram bot token",initialValue:e.telegramBotToken??"",validate:A("Bot token"),note:{title:"Bot token",body:`Telegram lets you create personal bots in ~30 seconds. Klaudius
170
+ personal pings). The next two prompts walk you through both.`}}))return{telegramBotToken:void 0,telegramChatId:void 0};let o=await t({type:"password",message:"Paste your Telegram bot token",initialValue:e.telegramBotToken??"",validate:C("Bot token"),note:{title:"Bot token",body:`Telegram lets you create personal bots in ~30 seconds. Klaudius
171
171
  uses a bot to send you private notifications.
172
172
 
173
173
  How to create one:
@@ -180,11 +180,16 @@ How to create one:
180
180
  'KlaudiusNotifierBot'). Each username has to be globally
181
181
  unique, so add a personal suffix if it's taken.
182
182
  6. BotFather replies with a token like
183
- '1234567890:AAEhBP_yourtokenhere\u2026'. Copy it and paste below.`}}),s=null,a=0,i=null;for(;s===null;){a++;let u=a>1;await t({type:"text",message:u?"Send another message to the bot, then press Enter to retry":"Send a message to your bot, then press Enter to detect chat ID",initialValue:"",placeholder:"(press Enter when you've sent the message)",validate:()=>{},note:{title:u?"Detect chat ID \u2014 retry":"Detect chat ID",body:u?`${i}
183
+ '1234567890:AAEhBP_yourtokenhere\u2026'. Copy it and paste below.`}}),r=null,a=0,s=null,u=null;for(;r===null;){a++;let i=a>1,l=u==="network"?`${s}
184
+
185
+ This is usually a network problem on your end (corporate WiFi or VPN
186
+ blocking Telegram, ISP DNS issue, a momentary drop). Check your
187
+ connection, try a different network if available, then press Enter
188
+ to retry.`:`${s}
184
189
 
185
190
  Make sure you sent a regular text message (e.g. 'hi'), not
186
191
  just a /start tap \u2014 /start alone doesn't always register.
187
- Then press Enter to retry.`:`Klaudius will auto-detect your chat ID by asking Telegram for
192
+ Then press Enter to retry.`;await t({type:"text",message:i?u==="network"?"Check your network, then press Enter to retry":"Send another message to the bot, then press Enter to retry":"Send a message to your bot, then press Enter to detect chat ID",initialValue:"",placeholder:"(press Enter when you've sent the message)",validate:()=>{},note:{title:i?"Detect chat ID \u2014 retry":"Detect chat ID",body:i?l:`Klaudius will auto-detect your chat ID by asking Telegram for
188
193
  any messages your bot has received. To make that work:
189
194
 
190
195
  1. In the Telegram app, search for the bot you just created
@@ -193,5 +198,5 @@ any messages your bot has received. To make that work:
193
198
  If the chat starts with a 'Start' button, tap that first,
194
199
  THEN send 'hi'. /start alone doesn't always show up.
195
200
  3. Press Enter below \u2014 Klaudius will detect the chat ID for
196
- you.`}});let r=await Q({botToken:o});if(r.ok&&r.chatId)s=r.chatId;else if(r.reason==="no_messages")i="No messages from you to your bot have been picked up yet.";else throw new Error(`Telegram chat ID auto-detect failed: ${r.message??"unknown error"}`)}return{telegramBotToken:o,telegramChatId:s}}},C=[ve,Ie,Te,ke,Ae,Ce,Ee,Oe,Me,Ne];import{Box as ne,Text as Re}from"ink";import{jsx as se,jsxs as Ke}from"react/jsx-runtime";var Be={pending:"\u25CB",current:"\u25C9",done:"\u2713",skipped:"\xB7"},Ve={pending:void 0,current:"cyan",done:"green",skipped:void 0};function ie({items:e}){return se(ne,{flexDirection:"column",width:28,flexShrink:0,children:e.map(t=>{let n=t.status==="skipped"||t.status==="pending";return se(ne,{children:Ke(Re,{color:Ve[t.status],bold:t.status==="current",dimColor:n&&t.status!=="current",children:[Be[t.status]," ",t.title]})},t.id)})})}import{useEffect as Fe,useState as re}from"react";import{Box as $,Text as D,useInput as _e}from"ink";import He from"ink-text-input";import{jsx as R,jsxs as j}from"react/jsx-runtime";function le({spec:e,masked:t,onSubmit:n,onBack:o}){let[s,a]=re(typeof e.initialValue=="string"?e.initialValue:""),[i,u]=re(null);Fe(()=>(g("TextPromptUI:mount",{masked:t,message:e.message.slice(0,60),initial:typeof e.initialValue=="string"?e.initialValue.slice(0,30):null}),()=>{g("TextPromptUI:unmount")}),[t,e.message,e.initialValue]),_e((l,p)=>{g("TextPromptUI:useInput fired",{input:l,ctrl:p.ctrl,escape:p.escape,return:p.return}),p.escape&&o()});let r=l=>{g("TextPromptUI:handleSubmit",{raw:l.slice(0,30)});let p=l;if(e.validate){let m=e.validate(p);if(m){g("TextPromptUI:validate failed",{err:m}),u(m);return}}u(null),n(p)};return j($,{flexDirection:"column",children:[j(D,{children:[R(D,{color:"cyan",children:"?"})," ",e.message]}),j($,{children:[R(D,{color:"green",children:"\u203A "}),R(He,{value:s,onChange:a,onSubmit:r,placeholder:e.placeholder,mask:t?"\u2022":void 0})]}),i&&R($,{marginTop:0,children:R(D,{color:"red",children:i})})]})}import{useState as Ue}from"react";import{Box as ce,Text as W,useInput as De}from"ink";import{jsx as ue,jsxs as q}from"react/jsx-runtime";function pe({spec:e,onSubmit:t,onBack:n}){let o=e.options??[],s=typeof e.initialValue=="string"?Math.max(0,o.findIndex(u=>u.value===e.initialValue)):0,[a,i]=Ue(s===-1?0:s);return De((u,r)=>{if(r.escape)n();else if(r.upArrow)i(l=>(l-1+o.length)%o.length);else if(r.downArrow)i(l=>(l+1)%o.length);else if(r.return){let l=o[a];l&&t(l.value)}}),q(ce,{flexDirection:"column",children:[q(W,{children:[ue(W,{color:"cyan",children:"?"})," ",e.message]}),ue(ce,{flexDirection:"column",marginTop:0,children:o.map((u,r)=>{let l=r===a;return q(W,{color:l?"cyan":void 0,bold:l,children:[l?"\u25CF ":"\u25CB ",u.label]},u.value)})})]})}import{useState as Ge}from"react";import{Box as de,Text as B,useInput as Le}from"ink";import{jsx as me,jsxs as V}from"react/jsx-runtime";function he({spec:e,onSubmit:t,onBack:n}){let[o,s]=Ge(typeof e.initialValue=="boolean"?e.initialValue:!0);return Le((a,i)=>{i.escape?n():i.leftArrow||i.rightArrow?s(u=>!u):a==="y"||a==="Y"?s(!0):a==="n"||a==="N"?s(!1):i.return&&t(o)}),V(de,{flexDirection:"column",children:[V(B,{children:[me(B,{color:"cyan",children:"?"})," ",e.message]}),V(de,{marginTop:0,children:[V(B,{color:o?"cyan":void 0,bold:o,children:[o?"\u25CF ":"\u25CB ","Yes"]}),me(B,{children:" "}),V(B,{color:o?void 0:"cyan",bold:!o,children:[o?"\u25CB ":"\u25CF ","No"]})]})]})}import{Box as ge,Text as fe}from"ink";import{jsx as Y,jsxs as ze}from"react/jsx-runtime";function X({title:e,body:t}){let n=t.split(`
197
- `);return ze(ge,{flexDirection:"column",borderStyle:"round",borderColor:"gray",paddingX:1,children:[Y(fe,{color:"cyan",bold:!0,children:e}),Y(ge,{flexDirection:"column",marginTop:0,children:n.map((o,s)=>Y(fe,{dimColor:!1,children:o.length===0?" ":o},s))})]})}import{jsx as d,jsxs as K}from"react/jsx-runtime";function Rt({defaults:e,onComplete:t,onlySection:n,banner:o}){let{exit:s}=We(),a=Z({...e}),i=Z(new Map),[,u]=T(0),r=()=>u(h=>h+1),[l,p]=T(null),[m,v]=T(null),[E,F]=T([]),[O,k]=T(null),[J,ye]=T(!1),[G,be]=T({current:0,total:0}),_=()=>{let h=i.current;for(let f of C){let z=f.isApplicable(a.current),c=h.get(f.id);z?n&&f.id!==n?c!=="current"&&h.set(f.id,"done"):(c==="skipped"||!c)&&h.set(f.id,"pending"):h.set(f.id,"skipped")}};i.current.size===0&&_();let L=Z(!1);$e(()=>{if(g("WizardApp:useEffect fired",{startedAlready:L.current}),L.current){g("WizardApp:useEffect returning early (startedRef true)");return}L.current=!0,g("WizardApp:orchestrator starting");let h=0,f=c=>{let w=++h;return g("orchestrator:promptFn called",{id:w,type:c.type,message:c.message.slice(0,60)}),k(null),new Promise((S,y)=>{v({id:w,spec:c,resolve:x=>{g("orchestrator:prompt resolved",{id:w,type:c.type,value:typeof x=="string"?x.slice(0,30):x}),S(x)},reject:x=>{g("orchestrator:prompt rejected",{id:w,reason:x.message}),y(x)}})})},z=c=>{F(w=>[...w,c])};return(async()=>{try{let c=n?C.filter(y=>y.id===n):C,w=()=>c.filter(y=>y.isApplicable(a.current)).length,S=0;for(;S<c.length;){let y=c[S];if(!y.isApplicable(a.current)){i.current.set(y.id,"skipped"),S++,_(),r();continue}F([]);for(let b of C)b.id===y.id?i.current.set(b.id,"current"):i.current.get(b.id)==="current"&&i.current.set(b.id,"pending");p(y.id);let x=c.slice(0,S+1).filter(b=>b.isApplicable(a.current)).length;be({current:x,total:w()}),r();try{let b=await y.execute(a.current,{prompt:f,info:z});a.current={...a.current,...b},i.current.set(y.id,"done"),_(),S++}catch(b){if(b instanceof N){i.current.set(y.id,"pending");let M=S-1;for(;M>=0&&!c[M].isApplicable(a.current);)M--;M<0||(S=M,i.current.set(c[S].id,"pending")),_(),r();continue}throw b}}g("orchestrator:all sections done, calling onComplete + exit"),ye(!0),v(null),p(null),r(),await new Promise(y=>setTimeout(y,200)),t(a.current),s()}catch(c){g("orchestrator:caught error",{name:c?.name,message:c?.message}),c instanceof U&&(s(),process.exit(0));let w=c instanceof Error?c.message:String(c);k(`Setup error: ${w}`),r()}})(),()=>{g("WizardApp:useEffect cleanup ran (component unmounting)")}},[s,t]),qe((h,f)=>{g("WizardApp:useInput fired",{input:h,ctrl:f.ctrl,escape:f.escape,return:f.return}),f.ctrl&&h==="c"&&(g("WizardApp:Ctrl+C \u2192 exiting"),s(),process.exit(0))});let we=je(()=>C.map(h=>({id:h.id,title:h.title,status:i.current.get(h.id)??"pending"})),[l,J,O]);return K(P,{flexDirection:"column",paddingY:1,children:[K(P,{marginBottom:1,children:[d(I,{bold:!0,color:"cyan",children:"Klaudius setup"}),d(I,{dimColor:!0,children:" \xB7 "}),d(I,{dimColor:!0,children:G.total>0?`Step ${G.current} of ${G.total}`:"Loading\u2026"})]}),K(P,{children:[d(ie,{items:we}),K(P,{flexDirection:"column",flexGrow:1,marginLeft:2,children:[o&&K(P,{marginBottom:1,borderStyle:"round",borderColor:"red",paddingX:1,flexDirection:"column",children:[d(I,{color:"red",bold:!0,children:"\u26A0 Heads up"}),d(I,{children:o})]}),E.map((h,f)=>d(P,{marginBottom:1,children:d(X,{title:h.title,body:h.body})},f)),m?.spec.note&&d(P,{marginBottom:1,children:d(X,{title:m.spec.note.title,body:m.spec.note.body})}),m&&Ye(m),O&&d(P,{marginTop:1,children:d(I,{color:"red",children:O})}),J&&d(P,{marginTop:1,children:d(I,{color:"green",bold:!0,children:"\u2713 Setup complete. Continuing\u2026"})})]})]}),d(P,{marginTop:1,children:d(I,{dimColor:!0,children:"\u21B5 confirm esc back to previous section ctrl-c cancel"})})]})}function Ye(e){let t=o=>e.resolve(o),n=()=>e.reject(new N);switch(e.spec.type){case"text":case"password":return d(le,{spec:e.spec,masked:e.spec.type==="password",onSubmit:t,onBack:n},e.id);case"select":return d(pe,{spec:e.spec,onSubmit:t,onBack:n},e.id);case"confirm":return d(he,{spec:e.spec,onSubmit:t,onBack:n},e.id)}}export{Rt as WizardApp};
201
+ you.`}});let c=await Q({botToken:o});if(c.ok&&c.chatId)r=c.chatId;else if(c.reason==="no_messages")s="No messages from you to your bot have been picked up yet.",u="no_messages";else if(c.reason==="network")s=c.message??"Could not reach Telegram.",u="network";else throw new Error(`Telegram chat ID auto-detect failed: ${c.message??"unknown error"}`)}return{telegramBotToken:o,telegramChatId:r}}},I=[ke,Ie,Te,Ae,Ce,Ee,Oe,Me,Ne,Re];import{Box as ae,Text as Be}from"ink";import{jsx as se,jsxs as Fe}from"react/jsx-runtime";var Ve={pending:"\u25CB",current:"\u25C9",done:"\u2713",skipped:"\xB7"},Ke={pending:void 0,current:"cyan",done:"green",skipped:void 0};function re({items:e}){return se(ae,{flexDirection:"column",width:28,flexShrink:0,children:e.map(t=>{let n=t.status==="skipped"||t.status==="pending";return se(ae,{children:Fe(Be,{color:Ke[t.status],bold:t.status==="current",dimColor:n&&t.status!=="current",children:[Ve[t.status]," ",t.title]})},t.id)})})}import{useEffect as _e,useState as ie}from"react";import{Box as z,Text as D,useInput as He}from"ink";import Ue from"ink-text-input";import{jsx as B,jsxs as W}from"react/jsx-runtime";function le({spec:e,masked:t,onSubmit:n,onBack:o}){let[r,a]=ie(typeof e.initialValue=="string"?e.initialValue:""),[s,u]=ie(null);_e(()=>(f("TextPromptUI:mount",{masked:t,message:e.message.slice(0,60),initial:typeof e.initialValue=="string"?e.initialValue.slice(0,30):null}),()=>{f("TextPromptUI:unmount")}),[t,e.message,e.initialValue]),He((l,c)=>{f("TextPromptUI:useInput fired",{input:l,ctrl:c.ctrl,escape:c.escape,return:c.return}),c.escape&&o()});let i=l=>{f("TextPromptUI:handleSubmit",{raw:l.slice(0,30)});let c=l;if(e.validate){let m=e.validate(c);if(m){f("TextPromptUI:validate failed",{err:m}),u(m);return}}u(null),n(c)};return W(z,{flexDirection:"column",children:[W(D,{children:[B(D,{color:"cyan",children:"?"})," ",e.message]}),W(z,{children:[B(D,{color:"green",children:"\u203A "}),B(Ue,{value:r,onChange:a,onSubmit:i,placeholder:e.placeholder,mask:t?"\u2022":void 0})]}),s&&B(z,{marginTop:0,children:B(D,{color:"red",children:s})})]})}import{useState as De}from"react";import{Box as ce,Text as j,useInput as Ge}from"ink";import{jsx as ue,jsxs as q}from"react/jsx-runtime";function pe({spec:e,onSubmit:t,onBack:n}){let o=e.options??[],r=typeof e.initialValue=="string"?Math.max(0,o.findIndex(u=>u.value===e.initialValue)):0,[a,s]=De(r===-1?0:r);return Ge((u,i)=>{if(i.escape)n();else if(i.upArrow)s(l=>(l-1+o.length)%o.length);else if(i.downArrow)s(l=>(l+1)%o.length);else if(i.return){let l=o[a];l&&t(l.value)}}),q(ce,{flexDirection:"column",children:[q(j,{children:[ue(j,{color:"cyan",children:"?"})," ",e.message]}),ue(ce,{flexDirection:"column",marginTop:0,children:o.map((u,i)=>{let l=i===a;return q(j,{color:l?"cyan":void 0,bold:l,children:[l?"\u25CF ":"\u25CB ",u.label]},u.value)})})]})}import{useState as Le}from"react";import{Box as de,Text as V,useInput as $e}from"ink";import{jsx as me,jsxs as K}from"react/jsx-runtime";function he({spec:e,onSubmit:t,onBack:n}){let[o,r]=Le(typeof e.initialValue=="boolean"?e.initialValue:!0);return $e((a,s)=>{s.escape?n():s.leftArrow||s.rightArrow?r(u=>!u):a==="y"||a==="Y"?r(!0):a==="n"||a==="N"?r(!1):s.return&&t(o)}),K(de,{flexDirection:"column",children:[K(V,{children:[me(V,{color:"cyan",children:"?"})," ",e.message]}),K(de,{marginTop:0,children:[K(V,{color:o?"cyan":void 0,bold:o,children:[o?"\u25CF ":"\u25CB ","Yes"]}),me(V,{children:" "}),K(V,{color:o?void 0:"cyan",bold:!o,children:[o?"\u25CB ":"\u25CF ","No"]})]})]})}import{Box as ge,Text as fe}from"ink";import{jsx as Y,jsxs as ze}from"react/jsx-runtime";function X({title:e,body:t}){let n=t.split(`
202
+ `);return ze(ge,{flexDirection:"column",borderStyle:"round",borderColor:"gray",paddingX:1,children:[Y(fe,{color:"cyan",bold:!0,children:e}),Y(ge,{flexDirection:"column",marginTop:0,children:n.map((o,r)=>Y(fe,{dimColor:!1,children:o.length===0?" ":o},r))})]})}import{jsx as d,jsxs as F}from"react/jsx-runtime";function Vt({defaults:e,onComplete:t,onlySection:n,banner:o}){let{exit:r}=qe(),a=Z({...e}),s=Z(new Map),[,u]=T(0),i=()=>u(h=>h+1),[l,c]=T(null),[m,x]=T(null),[E,_]=T([]),[O,A]=T(null),[J,ye]=T(!1),[G,be]=T({current:0,total:0}),M=()=>{let h=s.current;for(let y of I){let $=y.isApplicable(a.current),p=h.get(y.id);$?n&&y.id!==n?p!=="current"&&h.set(y.id,"done"):(p==="skipped"||!p)&&h.set(y.id,"pending"):h.set(y.id,"skipped")}};s.current.size===0&&M();let L=Z(!1);We(()=>{if(f("WizardApp:useEffect fired",{startedAlready:L.current}),L.current){f("WizardApp:useEffect returning early (startedRef true)");return}L.current=!0,f("WizardApp:orchestrator starting");let h=0,y=p=>{let S=++h;return f("orchestrator:promptFn called",{id:S,type:p.type,message:p.message.slice(0,60)}),A(null),new Promise((b,g)=>{x({id:S,spec:p,resolve:v=>{f("orchestrator:prompt resolved",{id:S,type:p.type,value:typeof v=="string"?v.slice(0,30):v}),b(v)},reject:v=>{f("orchestrator:prompt rejected",{id:S,reason:v.message}),g(v)}})})},$=p=>{_(S=>[...S,p])};return(async()=>{try{let p=n?I.filter(g=>g.id===n):I,S=()=>p.filter(g=>g.isApplicable(a.current)).length,b=0;for(;b<p.length;){let g=p[b];if(!g.isApplicable(a.current)){s.current.set(g.id,"skipped"),b++,M(),i();continue}_([]);for(let w of I)w.id===g.id?s.current.set(w.id,"current"):s.current.get(w.id)==="current"&&s.current.set(w.id,"pending");c(g.id);let v=p.slice(0,b+1).filter(w=>w.isApplicable(a.current)).length;be({current:v,total:S()}),i();try{let w=await g.execute(a.current,{prompt:y,info:$});a.current={...a.current,...w},s.current.set(g.id,"done"),M(),b++}catch(w){if(w instanceof R){if(n&&g.clearOnSkip){for(let Se of g.clearOnSkip)a.current[Se]=void 0;s.current.set(g.id,"skipped"),M(),b++;continue}if(n)continue;s.current.set(g.id,"pending");let N=b-1;for(;N>=0&&!p[N].isApplicable(a.current);)N--;N<0||(b=N,s.current.set(p[b].id,"pending")),M(),i();continue}throw w}}f("orchestrator:all sections done, calling onComplete + exit"),ye(!0),x(null),c(null),i(),await new Promise(g=>setTimeout(g,200)),t(a.current),r()}catch(p){f("orchestrator:caught error",{name:p?.name,message:p?.message}),p instanceof U&&(r(),process.exit(0));let S=p instanceof Error?p.message:String(p);A(`Setup error: ${S}`),i()}})(),()=>{f("WizardApp:useEffect cleanup ran (component unmounting)")}},[r,t]),Ye((h,y)=>{f("WizardApp:useInput fired",{input:h,ctrl:y.ctrl,escape:y.escape,return:y.return}),y.ctrl&&h==="c"&&(f("WizardApp:Ctrl+C \u2192 exiting"),r(),process.exit(0))});let we=je(()=>I.map(h=>({id:h.id,title:h.title,status:s.current.get(h.id)??"pending"})),[l,J,O]);return F(P,{flexDirection:"column",paddingY:1,children:[F(P,{marginBottom:1,children:[d(k,{bold:!0,color:"cyan",children:"Klaudius setup"}),d(k,{dimColor:!0,children:" \xB7 "}),d(k,{dimColor:!0,children:G.total>0?`Step ${G.current} of ${G.total}`:"Loading\u2026"})]}),F(P,{children:[d(re,{items:we}),F(P,{flexDirection:"column",flexGrow:1,marginLeft:2,children:[o&&F(P,{marginBottom:1,borderStyle:"round",borderColor:"red",paddingX:1,flexDirection:"column",children:[d(k,{color:"red",bold:!0,children:"\u26A0 Heads up"}),d(k,{children:o})]}),E.map((h,y)=>d(P,{marginBottom:1,children:d(X,{title:h.title,body:h.body})},y)),m?.spec.note&&d(P,{marginBottom:1,children:d(X,{title:m.spec.note.title,body:m.spec.note.body})}),m&&Ze(m),O&&d(P,{marginTop:1,children:d(k,{color:"red",children:O})}),J&&d(P,{marginTop:1,children:d(k,{color:"green",bold:!0,children:"\u2713 Setup complete. Continuing\u2026"})})]})]}),d(P,{marginTop:1,children:d(k,{dimColor:!0,children:Xe(n,l)})})]})}function Xe(e,t){if(e){let n=I.find(o=>o.id===t);return n?.clearOnSkip&&n.clearOnSkip.length>0?"\u21B5 confirm esc skip this section ctrl-c cancel":"\u21B5 confirm ctrl-c cancel"}return"\u21B5 confirm esc back to previous section ctrl-c cancel"}function Ze(e){let t=o=>e.resolve(o),n=()=>e.reject(new R);switch(e.spec.type){case"text":case"password":return d(le,{spec:e.spec,masked:e.spec.type==="password",onSubmit:t,onBack:n},e.id);case"select":return d(pe,{spec:e.spec,onSubmit:t,onBack:n},e.id);case"confirm":return d(he,{spec:e.spec,onSubmit:t,onBack:n},e.id)}}export{Vt as WizardApp};
package/dist/bin.js CHANGED
@@ -1,47 +1,59 @@
1
1
  #!/usr/bin/env node
2
- import{a as _,b as F,c as V,e as G}from"./chunk-GI7NTKNT.js";import{a as U,e as W}from"./chunk-47ODNABO.js";import{Command as yt}from"commander";import{mkdir as ye,readdir as et,readFile as he,stat as tt,writeFile as ge}from"fs/promises";import{resolve as st,isAbsolute as it,join as x}from"path";import*as d from"@clack/prompts";import m from"picocolors";import be from"react";import{render as Re}from"ink";async function y(e={},s={}){if(!process.stdin.isTTY)throw new Error("Klaudius's interactive wizard requires a terminal (TTY). Run `npx klaudius init` directly from your terminal \u2014 not through a pipe, file redirect, or non-interactive shell context.");let{WizardApp:n}=await import("./WizardApp-ETYSKVKU.js"),t=null;_("runWizard:start",{onlySection:s.onlySection,hasDefaults:Object.keys(e).length>0}),V(),await new Promise(l=>setImmediate(l)),await new Promise(l=>setTimeout(l,0)),_("runWizard:event loop drained, mounting Ink");let a=Re(be.createElement(n,{defaults:e,onlySection:s.onlySection,banner:s.banner,onComplete:l=>{_("runWizard:onComplete called",{keys:Object.keys(l).length}),t=l}}),{stdout:process.stdout,stderr:process.stderr,stdin:process.stdin,exitOnCtrlC:!1});if(_("runWizard:render returned, awaiting exit"),await a.waitUntilExit(),_("runWizard:waitUntilExit resolved",{resultIsNull:t===null}),!t)throw new Error(`Wizard exited without completing. Diagnostic log: ${F}
3
- Please share that file so we can trace what happened.`);return t}function C(e){let s=[],n=a=>{s.push(""),s.push("# "+"=".repeat(60)),s.push(`# ${a}`),s.push("# "+"=".repeat(60)),s.push("")},t=(a,l)=>{if(l==null||l===""){s.push(`# ${a}=`);return}let c=String(l);if(/[\s"'`$\\#&|<>;()*?\[\]{}!~]/.test(c)){let r=c.replace(/\\/g,"\\\\").replace(/"/g,'\\"').replace(/\$/g,"\\$").replace(/`/g,"\\`");s.push(`${a}="${r}"`)}else s.push(`${a}=${c}`)};return s.push("# Klaudius configuration"),s.push("# Generated by `npx klaudius init`. Edit freely."),s.push("# NEVER commit this file. .gitignore already excludes it."),n("Klaudius licence"),t("KLAUDIUS_LICENSE_KEY",e.licenseKey),n("Pricing"),t("PRICING",e.pricing),t("PRICING_TERMS",e.pricingTerms),t("PRICING_MONTHLY",e.pricingMonthly),n("Automated outreach features"),t("OUTREACH_ENABLED",e.outreachEnabled?"true":"false"),t("OPERATOR_NAME",e.name),t("SIGNATURE",e.signature),t("OUTREACH_CHANNELS",e.outreachChannels?.join(",")),t("OUTREACH_PRIORITY",e.outreachPriority),n("Email outreach"),t("EMAIL_ADDRESS",e.emailAddress),t("EMAIL_PASSWORD",e.emailPassword),t("EMAIL_FROM_NAME",e.emailFromName),t("EMAIL_SMTP_HOST",e.emailSmtpHost),t("EMAIL_SMTP_PORT",e.emailSmtpPort),t("EMAIL_IMAP_HOST",e.emailImapHost),t("EMAIL_IMAP_PORT",e.emailImapPort),t("EMAIL_SENT_FOLDER",e.emailSentFolder),t("EMAIL_INBOX_FOLDER",e.emailInboxFolder),n("SMS outreach"),t("SMS_PROVIDER",e.smsProvider),t("TEST_PHONE",e.testPhone),t("TWILIO_ACCOUNT_SID",e.twilioAccountSid),t("TWILIO_AUTH_TOKEN",e.twilioAuthToken),t("TWILIO_PHONE_NUMBER",e.twilioPhoneNumber),n("Supabase (pipeline state)"),t("SUPABASE_URL",e.supabaseUrl),t("SUPABASE_SERVICE_KEY",e.supabaseServiceKey),t("SUPABASE_PAT",e.supabasePat),t("SUPABASE_PROJECT_REF",e.supabaseProjectRef),n("Vercel (deployment)"),t("VERCEL_TOKEN",e.vercelToken),n("Search APIs"),t("GOOGLE_PLACES_API_KEY",e.googlePlacesApiKey),t("SERPAPI_KEY",e.serpapiKey),n("Telegram notifications (optional)"),t("TELEGRAM_BOT_TOKEN",e.telegramBotToken),t("TELEGRAM_CHAT_ID",e.telegramChatId),n("Rate limiting (sensible defaults)"),t("REQUESTS_PER_MINUTE",10),t("REQUEST_DELAY",2),s.join(`
2
+ import{a as x,b as ue,c as pe,d as T,f as me,g as fe}from"./chunk-SZ4KCTSL.js";import{a as ne,e as he}from"./chunk-47ODNABO.js";import{Command as us}from"commander";import{mkdir as je,readdir as Ut,readFile as De,stat as Dt,writeFile as Ke}from"fs/promises";import{resolve as Kt,isAbsolute as jt,join as Z}from"path";import*as g from"@clack/prompts";import m from"picocolors";import Ze from"react";import{render as et}from"ink";async function b(e={},s={}){if(!process.stdin.isTTY)throw new Error("Klaudius's interactive wizard requires a terminal (TTY). Run `npx klaudius init` directly from your terminal \u2014 not through a pipe, file redirect, or non-interactive shell context.");let{WizardApp:n}=await import("./WizardApp-B6HU5YMU.js"),t=null;x("runWizard:start",{onlySection:s.onlySection,hasDefaults:Object.keys(e).length>0}),pe(),await new Promise(a=>setImmediate(a)),await new Promise(a=>setTimeout(a,0)),x("runWizard:event loop drained, mounting Ink");let o=et(Ze.createElement(n,{defaults:e,onlySection:s.onlySection,banner:s.banner,onComplete:a=>{x("runWizard:onComplete called",{keys:Object.keys(a).length}),t=a}}),{stdout:process.stdout,stderr:process.stderr,stdin:process.stdin,exitOnCtrlC:!1});if(x("runWizard:render returned, awaiting exit"),await o.waitUntilExit(),x("runWizard:waitUntilExit resolved",{resultIsNull:t===null}),!t)throw new Error(`Wizard exited without completing. Diagnostic log: ${ue}
3
+ Please share that file so we can trace what happened.`);return t}function B(e){let s=[],n=o=>{s.push(""),s.push("# "+"=".repeat(60)),s.push(`# ${o}`),s.push("# "+"=".repeat(60)),s.push("")},t=(o,a)=>{if(a==null||a===""){s.push(`# ${o}=`);return}let l=String(a);if(/[\s"'`$\\#&|<>;()*?\[\]{}!~]/.test(l)){let c=l.replace(/\\/g,"\\\\").replace(/"/g,'\\"').replace(/\$/g,"\\$").replace(/`/g,"\\`");s.push(`${o}="${c}"`)}else s.push(`${o}=${l}`)};return s.push("# Klaudius configuration"),s.push("# Generated by `npx klaudius init`. Edit freely."),s.push("# NEVER commit this file. .gitignore already excludes it."),n("Klaudius licence"),t("KLAUDIUS_LICENSE_KEY",e.licenseKey),n("Pricing"),t("PRICING",e.pricing),t("PRICING_TERMS",e.pricingTerms),t("PRICING_MONTHLY",e.pricingMonthly),n("Automated outreach features"),t("OUTREACH_ENABLED",e.outreachEnabled?"true":"false"),t("OPERATOR_NAME",e.name),t("SIGNATURE",e.signature),t("OUTREACH_CHANNELS",e.outreachChannels?.join(",")),t("OUTREACH_PRIORITY",e.outreachPriority),n("Email outreach"),t("EMAIL_ADDRESS",e.emailAddress),t("EMAIL_PASSWORD",e.emailPassword),t("EMAIL_FROM_NAME",e.emailFromName),t("EMAIL_SMTP_HOST",e.emailSmtpHost),t("EMAIL_SMTP_PORT",e.emailSmtpPort),t("EMAIL_IMAP_HOST",e.emailImapHost),t("EMAIL_IMAP_PORT",e.emailImapPort),t("EMAIL_SENT_FOLDER",e.emailSentFolder),t("EMAIL_INBOX_FOLDER",e.emailInboxFolder),n("SMS outreach"),t("SMS_PROVIDER",e.smsProvider),t("TEST_PHONE",e.testPhone),t("TWILIO_ACCOUNT_SID",e.twilioAccountSid),t("TWILIO_AUTH_TOKEN",e.twilioAuthToken),t("TWILIO_PHONE_NUMBER",e.twilioPhoneNumber),n("Supabase (pipeline state)"),t("SUPABASE_URL",e.supabaseUrl),t("SUPABASE_SERVICE_KEY",e.supabaseServiceKey),t("SUPABASE_PAT",e.supabasePat),t("SUPABASE_PROJECT_REF",e.supabaseProjectRef),n("Vercel (deployment)"),t("VERCEL_TOKEN",e.vercelToken),t("VERCEL_SCOPE",e.vercelScope??""),n("Search APIs"),t("GOOGLE_PLACES_API_KEY",e.googlePlacesApiKey),t("SERPAPI_KEY",e.serpapiKey),n("Telegram notifications (optional)"),t("TELEGRAM_BOT_TOKEN",e.telegramBotToken),t("TELEGRAM_CHAT_ID",e.telegramChatId),n("Rate limiting (sensible defaults)"),t("REQUESTS_PER_MINUTE",10),t("REQUEST_DELAY",2),s.join(`
4
4
  `)+`
5
- `}function v(e){let s={mcpServers:{supabase:{command:"npx",args:["-y","@supabase/mcp-server-supabase@latest",`--project-ref=${e.supabaseProjectRef}`],env:{SUPABASE_ACCESS_TOKEN:e.supabasePat}}}};return JSON.stringify(s,null,2)+`
6
- `}function Ce(e,s){if(/[\s"'`$\\#&|<>;()*?\[\]{}!~]/.test(s)){let n=s.replace(/\\/g,"\\\\").replace(/"/g,'\\"').replace(/\$/g,"\\$").replace(/`/g,"\\`");return`${e}="${n}"`}return`${e}=${s}`}function Y(e,s,n){let t=n.filter(c=>!(c.key in s));if(t.length===0)return null;let a=e.endsWith(`
5
+ `}function F(e){let s={mcpServers:{supabase:{command:"npx",args:["-y","@supabase/mcp-server-supabase@latest",`--project-ref=${e.supabaseProjectRef}`],env:{SUPABASE_ACCESS_TOKEN:e.supabasePat,NODE_OPTIONS:"--use-system-ca"}}}};return JSON.stringify(s,null,2)+`
6
+ `}function tt(e,s){if(/[\s"'`$\\#&|<>;()*?\[\]{}!~]/.test(s)){let n=s.replace(/\\/g,"\\\\").replace(/"/g,'\\"').replace(/\$/g,"\\$").replace(/`/g,"\\`");return`${e}="${n}"`}return`${e}=${s}`}function ge(e,s,n){let t=n.filter(l=>!(l.key in s));if(t.length===0)return null;let o=e.endsWith(`
7
7
  `)?"":`
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(`
8
+ `,a=["","# "+"=".repeat(60),"# Backfilled by `klaudius update` (new template env keys)","# "+"=".repeat(60)];for(let l of t)a.push(""),l.comment&&a.push(`# ${l.comment}`),a.push(tt(l.key,l.value));return e+o+a.join(`
9
9
  `)+`
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.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
- `,"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.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:
10
+ `}function st(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 v(e){let s={};for(let n of e.split(`
11
+ `)){let t=n.trim();if(!t||t.startsWith("#"))continue;let o=t.indexOf("=");if(o===-1)continue;let a=t.slice(0,o).trim(),l=st(t.slice(o+1).trim());s[a]=l}return s}function ke(e){let s=a=>{let l=e[a];if(!l)return;let c=parseInt(l,10);return isNaN(c)?void 0:c},n=e.SMS_PROVIDER,t=e.OUTREACH_CHANNELS?.split(",").map(a=>a.trim()).filter(a=>a==="email"||a==="sms"),o=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:o==="email"||o==="sms"?o: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,vercelScope:e.VERCEL_SCOPE||void 0,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 ie(e){let s=P=>{let p=e[P];if(!p)return;let E=parseInt(p,10);return isNaN(E)?void 0:E},n=e.SUPABASE_URL,t=e.SUPABASE_SERVICE_KEY,o=e.SUPABASE_PAT,a=e.SUPABASE_PROJECT_REF,l=e.PRICING,c=e.PRICING_TERMS||"one-off, no monthly fees",h=e.VERCEL_TOKEN;if(!l||!n||!t||!o||!a||!h)return null;let r=e.SMS_PROVIDER,w=e.OUTREACH_CHANNELS?.split(",").map(P=>P.trim()).filter(P=>P==="email"||P==="sms"),k=e.OUTREACH_PRIORITY;return{licenseKey:e.KLAUDIUS_LICENSE_KEY,pricing:l,pricingTerms:c,pricingMonthly:e.PRICING_MONTHLY||void 0,outreachEnabled:e.OUTREACH_ENABLED?.toLowerCase()==="true",name:e.OPERATOR_NAME,signature:e.SIGNATURE,outreachChannels:w&&w.length>0?w:void 0,outreachPriority:k==="email"||k==="sms"?k: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:r==="imessage"||r==="twilio"?r:void 0,testPhone:e.TEST_PHONE,twilioAccountSid:e.TWILIO_ACCOUNT_SID,twilioAuthToken:e.TWILIO_AUTH_TOKEN,twilioPhoneNumber:e.TWILIO_PHONE_NUMBER,vercelToken:h,vercelScope:e.VERCEL_SCOPE||void 0,supabaseUrl:n,supabaseServiceKey:t,supabasePat:o,supabaseProjectRef:a,googlePlacesApiKey:e.GOOGLE_PLACES_API_KEY,serpapiKey:e.SERPAPI_KEY,telegramBotToken:e.TELEGRAM_BOT_TOKEN,telegramChatId:e.TELEGRAM_CHAT_ID}}import{mkdir as nt,rm as it}from"fs/promises";import{Readable as ot}from"stream";import*as we from"tar";var rt="https://klaudius.dev/api/template";function at(e){return ot.fromWeb(e)}async function G(e){let s=process.env.KLAUDIUS_TEMPLATE_URL??rt,n="0.7.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(l){throw new Error(`Could not reach klaudius.dev to download the template: ${T(l)}`)}if(!t.ok){let l="";try{let c=await t.json();l=c.reason?` (${c.reason})`:""}catch{}throw new Error(`Template download failed: HTTP ${t.status}${l}`)}if(!t.body)throw new Error("Template download succeeded but response had no body");let o=t.headers.get("x-klaudius-tier");await nt(e.destDir,{recursive:!0});let a=0;try{await new Promise((l,c)=>{let h=we.extract({cwd:e.destDir,strict:!0,onReadEntry:()=>{a++}});h.on("finish",l),h.on("error",c),at(t.body).on("error",c).pipe(h)})}catch(l){throw await it(e.destDir,{recursive:!0,force:!0}).catch(()=>{}),new Error(`Failed to extract template archive: ${l.message}`)}return{filesExtracted:a,tier:o}}import{mkdir as lt,readFile as ct,writeFile as dt,access as ut}from"fs/promises";import{dirname as pt,join as ye}from"path";var mt=".klaudius",Ee=`${mt}/manifest.json`;async function Pe(e){let s=ye(e,Ee);try{await ut(s)}catch{return null}let n=await ct(s,"utf-8");return JSON.parse(n)}async function Y(e,s){let n=ye(e,Ee);await lt(pt(n),{recursive:!0}),await dt(n,JSON.stringify(s,null,2)+`
12
+ `,"utf-8")}function Se(e,s){let n=new Date().toISOString();return{version:1,templateVersion:e,installedAt:n,updatedAt:n,files:s}}import{createHash as ft}from"crypto";import{hostname as be,userInfo as ht,platform as Ae,arch as Te}from"os";import O from"picocolors";var i={info:e=>console.log(O.cyan("\u2139"),e),success:e=>console.log(O.green("\u2713"),e),warn:e=>console.log(O.yellow("\u26A0"),e),error:e=>console.error(O.red("\u2717"),e),step:e=>console.log(O.dim("\u2192"),e),blank:()=>console.log(""),heading:e=>console.log(`
13
+ `+O.bold(O.cyan(e))),detail:e=>console.log(O.dim(" "+e))};var gt="https://klaudius.dev",kt="/api/licenses/validate",wt="0.7.0";async function _e(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??gt).replace(/\/$/,"")+kt,o=H(),a;try{a=await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({license_key:s,machine_id:o,hostname:be(),os:`${Ae()}-${Te()}`,cli_version:wt})})}catch(c){return{ok:!1,reason:`Could not reach the Klaudius license server (${T(c)}). Check your internet connection or try again later.`}}let l;try{l=await a.json()}catch{return{ok:!1,reason:`License server returned an unparseable response (HTTP ${a.status}).`}}return l.ok?{ok:!0,tier:l.tier??"core"}:{ok:!1,reason:l.reason??`License rejected (HTTP ${a.status}).`}}function H(){let e=[be(),ht().username,Ae(),Te()].join("|");return ft("sha256").update(e).digest("hex").slice(0,32)}import{readFile as yt}from"fs/promises";import{join as Et}from"path";async function $(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: ${T(s)}`}}}async function Ie(e){let s;try{s=await yt(Et(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 W(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: ${T(s)}`}}}async function z(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??"",o=n.error?.status??"";return s.status===400&&(o==="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: ${T(s)}`}}}import Pt from"nodemailer";import{ImapFlow as St}from"imapflow";async function ve(e){let s=Pt.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:Re(t,"SMTP")}}finally{s.close()}let n=new St({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:Re(t,"IMAP")}}return{ok:!0}}function Re(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
- \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:
17
- ${p}
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 o=await t.json();return o.status&&o.status!=="active"?{ok:!1,message:`Twilio account is in '${o.status}' state. Check the console for any holds before sending SMS.`}:{ok:!0,friendlyName:o.friendly_name}}catch(t){return{ok:!1,message:`Could not reach Twilio: ${T(t)}`}}}import{spawn as bt}from"child_process";async function Oe(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 At(n);if(t.code===0)return{ok:!0};let o=t.stderr.toLowerCase();return o.includes("can't get application")||o.includes("isn't running")||o.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."}:o.includes("(-1728)")||o.includes("not found")||o.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.`}:o.includes("(-1719)")||o.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 At(e){return new Promise(s=>{let n=bt("osascript",["-e",e],{stdio:["ignore","ignore","pipe"]}),t="";n.stderr.on("data",o=>{t+=o.toString()}),n.on("close",o=>s({code:o??1,stderr:t})),n.on("error",o=>s({code:1,stderr:o.message}))})}import{spawn as Ne}from"child_process";function Tt(){let e=process.env.NODE_OPTIONS?.trim(),s=e?`${e} --use-system-ca`:"--use-system-ca";return{...process.env,NODE_OPTIONS:s}}var M=2,xe=2e3,_t=[/ECONNRESET/,/ETIMEDOUT/,/ENOTFOUND/,/EAI_AGAIN/,/socket hang up/i,/Connection reset/i,/network timeout/i,/could not resolve host/i];function $e(e,s){let n=e+`
17
+ `+s;return _t.some(t=>t.test(n))}function Me(e){return new Promise(s=>setTimeout(s,e))}function It(e,s,n){return new Promise(t=>{let o=Ne(e,s,{cwd:n.cwd,stdio:["ignore","pipe","pipe"],shell:!1,env:n.env??process.env}),a="",l="";o.stdout?.on("data",c=>{let h=c.toString();a+=h,process.stdout.write(h)}),o.stderr?.on("data",c=>{let h=c.toString();l+=h,process.stderr.write(h)}),o.on("close",c=>t({code:c??1,stdout:a,stderr:l})),o.on("error",c=>t({code:1,stdout:a,stderr:l+c.message}))})}async function oe(e,s,n){let t={code:0,stdout:"",stderr:""};for(let o=0;o<=M;o++){if(t=await It(e,s,{cwd:n.cwd,env:n.env}),t.code===0||!$e(t.stderr,t.stdout))return t;o<M&&(i.detail(`${n.label} failed with a transient network error \u2014 retrying (${o+2}/${M+1})\u2026`),await Me(xe))}return t}function Rt(e,s,n){return new Promise(t=>{let o=Ne(e,s,{cwd:n.cwd,stdio:["ignore","pipe","pipe"],shell:!1}),a="",l="";o.stdout?.on("data",c=>{a+=c.toString()}),o.stderr?.on("data",c=>{l+=c.toString()}),o.on("close",c=>t({code:c??1,stdout:a,stderr:l})),o.on("error",c=>t({code:1,stdout:a,stderr:l+c.message}))})}var vt=[{pattern:/UNABLE_TO_VERIFY_LEAF_SIGNATURE|unable to verify the first certificate/i,hint:"TLS-handshake error \u2014 your machine appears to be intercepting HTTPS (antivirus, corporate proxy, or VPN re-signing traffic with its own root CA). We work around this with NODE_OPTIONS=--use-system-ca, which needs Node 22.15+. Run `node --version` to verify, upgrade if older, then re-run setup."},{pattern:/Microsoft Visual C\+\+ 14\.0 or greater is required/i,hint:"Python tried to compile a package from source and couldn't find the Microsoft C++ Build Tools. The latest Klaudius pins supabase<2.26 to skip the offending dependency entirely. Update with `npm install -g klaudius@latest` and re-run init \u2014 that should bypass the source build."}];function Ct(e){let s=`${e.stderr??""}
18
+ ${e.stdout??""}`;return vt.find(t=>t.pattern.test(s))?.hint}async function Ot(e){let s=["supabase<2.26","python-dotenv","requests","httpx","Pillow"],n=["install","--quiet","--user"],t=async a=>{let l={code:0,stdout:"",stderr:""};for(let c=0;c<=M;c++){if(l=await Rt("pip3",a,{cwd:e}),l.code===0||!$e(l.stderr,l.stdout))return l;c<M&&(i.detail(`pip install hit a transient network error \u2014 retrying (${c+2}/${M+1})\u2026`),await Me(xe))}return l},o=await t([...n,"--break-system-packages",...s]);return o.code!==0&&/no such option:.*break-system-packages/i.test(o.stderr)&&(i.detail("Older pip detected; retrying without --break-system-packages"),o=await t([...n,...s])),o.code!==0&&(o.stderr.trim()&&process.stderr.write(o.stderr),o.stdout.trim()&&process.stdout.write(o.stdout)),o}async function Le(e){let s={npm:"skipped",pip:"skipped",playwright:"skipped",patchright:"skipped"},n=Tt(),t=(h,r)=>{let w=Ct(r);w&&((s.hints??={})[h]=w)};i.step("Running npm install (Next.js, Playwright JS, etc.)...");let o=await oe("npm",["install"],{cwd:e,label:"npm install",env:n});s.npm=o.code===0?"ok":"failed",o.code!==0&&t("npm",o),i.step("Installing Python packages (supabase, dotenv, requests, httpx, Pillow)...");let a=await Ot(e);s.pip=a.code===0?"ok":"failed",a.code!==0&&t("pip",a),i.step("Installing Playwright browser (chromium ~200MB; one-time)...");let l=await oe("npx",["playwright","install","chromium"],{cwd:e,label:"playwright install",env:n});s.playwright=l.code===0?"ok":"failed",l.code!==0&&t("playwright",l),i.step("Installing Patchright browser (chromium ~200MB; one-time)...");let c=await oe("npx",["patchright","install","chromium"],{cwd:e,label:"patchright install",env:n});return s.patchright=c.code===0?"ok":"failed",c.code!==0&&t("patchright",c),s}import{spawn as Nt}from"child_process";var q=3,re=10;async function J(){let e=await xt("python3",["--version"]);if(e.code!==0)return{ok:!1,message:"python3 is not installed (or not on PATH). Install Python 3.10 or newer from https://www.python.org/downloads/macos/ (macOS) or https://www.python.org/downloads/ (other)."};let s=(e.stdout+e.stderr).trim(),n=/Python\s+(\d+)\.(\d+)(?:\.(\d+))?/.exec(s);if(!n)return{ok:!1,message:`Could not parse the python3 version string ("${s}"). Install Python 3.10 or newer from https://www.python.org/downloads/.`};let t=parseInt(n[1],10),o=parseInt(n[2],10),a=n[3]?`${t}.${o}.${n[3]}`:`${t}.${o}`;return t<q||t===q&&o<re?{ok:!1,version:a,major:t,minor:o,message:`Your installed Python is ${a}. Klaudius needs Python ${q}.${re} or newer (the pipeline depends on packages that only ship wheels for ${q}.${re}+). Install a newer Python from https://www.python.org/downloads/macos/ \u2014 it lives alongside your existing Python, nothing breaks. After installing, close every Terminal window, open a new one, then re-run klaudius.`}:{ok:!0,version:a,major:t,minor:o}}function xt(e,s){return new Promise(n=>{let t=Nt(e,s,{stdio:["ignore","pipe","pipe"],shell:!1}),o="",a="";t.stdout?.on("data",l=>{o+=l.toString()}),t.stderr?.on("data",l=>{a+=l.toString()}),t.on("close",l=>n({code:l??1,stdout:o,stderr:a})),t.on("error",()=>n({code:1,stdout:o,stderr:a}))})}function X(){let e=process.version,s=/^v(\d+)\.(\d+)\.(\d+)/.exec(e);if(!s)return{ok:!1,message:`Could not parse the Node.js version string ("${e}"). Klaudius requires Node.js 22.15 or newer \u2014 install from https://nodejs.org/.`};let n=parseInt(s[1],10),t=parseInt(s[2],10),o=parseInt(s[3],10),a=`${n}.${t}.${o}`;return n<22||n===22&&t<15?{ok:!1,version:a,major:n,minor:t,message:`Your installed Node.js is ${a}. Klaudius requires Node.js 22.15 or newer (the dependency installer relies on the --use-system-ca flag, which was added in 22.15 to handle Windows machines where antivirus or corporate-proxy software MITMs HTTPS). Install a newer Node.js from https://nodejs.org/ \u2014 after installing, close every Terminal window, open a new one, then re-run klaudius.`}:{ok:!0,version:a,major:n,minor:t}}import{mkdir as $t,readFile as Mt,writeFile as Lt}from"fs/promises";import{join as Ue}from"path";async function Q(e,s){let n=Ue(e,".claude");await $t(n,{recursive:!0});let t=Ue(n,"settings.json"),o={};try{let h=await Mt(t,"utf-8"),r=JSON.parse(h);r&&typeof r=="object"&&!Array.isArray(r)&&(o=r)}catch{}let a=o.enabledMcpjsonServers,l=Array.isArray(a)?a.filter(h=>typeof h=="string"):[];l.includes(s)||l.push(s);let c={...o,enabledMcpjsonServers:l};await Lt(t,JSON.stringify(c,null,2)+`
19
+ `,"utf-8")}var A=5;function L(e,s){i.blank(),i.error(`${e} validation failed ${A} times in a row. Pausing setup so you don't get stuck in a loop.`),i.detail(`Last error: ${s}`),i.blank(),i.detail("Your credentials are saved in `.env`. Two ways forward:"),i.detail(" \u2022 Open `.env` in any text editor, fix the value, re-run `klaudius init` and pick Resume."),i.detail(" \u2022 Or email hello@klaudius.dev with the error above and we'll dig in."),i.blank(),process.exit(1)}async function He(e){let s=e.target??".",n=jt(s)?s:Kt(process.cwd(),s),t=await Ht(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 o=await Bt(e.license),a=g.spinner();a.start("Validating license");let l=await _e(o);l.ok||(a.stop(`License invalid: ${l.reason??"unknown"}`,2),i.detail("If you believe this is in error, email hello@klaudius.dev with your purchase details."),process.exit(1)),a.stop(`License valid (${l.tier??"core"} tier)`);let c=await J();c.ok||(i.error(c.message??"Python version check failed."),i.blank(),i.detail("Once you've installed Python 3.10+, re-run `klaudius init` in this folder."),process.exit(1));let h=X();h.ok||(i.error(h.message??"Node.js version check failed."),i.blank(),i.detail("Once you've installed Node.js 22.15+, re-run `klaudius init` in this folder."),process.exit(1));let r,w=!1;if(t.kind==="partial")if(await Vt(n)==="resume"){let f=await De(t.envPath,"utf-8"),d=ie(v(f));d?(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=d,w=!0):(i.warn("Existing .env was missing required fields \u2014 starting wizard fresh."),r=await b())}else{let f=await De(t.envPath,"utf-8"),d=ie(v(f));r=await b(d??{})}else r=await b();r.licenseKey=o,await C(n,r),w||(i.success("Saved .env"),i.success("Saved .mcp.json")),i.blank();for(let u=1;;u++){let f=g.spinner();f.start("Validating Supabase credentials");let d=await $({url:r.supabaseUrl,serviceKey:r.supabaseServiceKey});if(d.ok){f.stop("Supabase credentials accepted");break}f.stop(d.message??"Supabase check failed",2),u>=A&&L("Supabase",d.message??"Supabase check failed");let y=d.message??"Supabase check failed";i.blank(),i.warn(`Bouncing you back into the Supabase section so you can fix it in place. (attempt ${u+1} of ${A})`),i.detail("(Press Enter through any field to keep its current value. Ctrl+C to exit.)"),i.blank(),r=await b(r,{onlySection:"supabase",banner:`Supabase rejected your credentials:
20
+ ${y}
18
21
 
19
22
  Re-enter the URL, secret key, or PAT below. Press Enter on any
20
- field to keep what's already there.`}),await S(n,r)}for(i.blank();;){let u=d.spinner();u.start("Validating Vercel token");let o=await re(r.vercelToken);if(o.ok){u.stop(o.username?`Vercel token accepted (signed in as ${o.username})`:"Vercel token accepted");break}u.stop(o.message??"Vercel check failed",2);let p=o.message??"Vercel check failed";i.blank(),i.warn("Bouncing you back into the Vercel section so you can fix it in place."),i.detail("(Press Enter on the field to keep its current value. Ctrl+C to exit.)"),i.blank(),r=await y(r,{onlySection:"vercel",banner:`Vercel rejected the token:
21
- ${p}
23
+ field to keep what's already there.`}),await C(n,r)}i.blank();for(let u=1;;u++){let f=g.spinner();f.start("Validating Vercel token");let d=await W(r.vercelToken);if(d.ok){f.stop(d.username?`Vercel token accepted (signed in as ${d.username})`:"Vercel token accepted");break}f.stop(d.message??"Vercel check failed",2),u>=A&&L("Vercel",d.message??"Vercel check failed");let y=d.message??"Vercel check failed";i.blank(),i.warn(`Bouncing you back into the Vercel section so you can fix it in place. (attempt ${u+1} of ${A})`),i.detail("(Press Enter on the field to keep its current value. Ctrl+C to exit.)"),i.blank(),r=await b(r,{onlySection:"vercel",banner:`Vercel rejected the token:
24
+ ${y}
22
25
 
23
26
  Generate a new token at https://vercel.com/account/tokens and
24
- paste it below. Press Enter to keep the current value.`}),await S(n,r)}for(i.blank();;){let u=d.spinner();u.start("Validating Google Places API key");let o=await ae(r.googlePlacesApiKey??"");if(o.ok){u.stop("Google Places API key accepted");break}u.stop(o.message??"Google Places check failed",2);let p=o.message??"Google Places check failed";i.blank(),i.warn("Bouncing you back into the Google Places section so you can fix it in place."),i.detail("(Press Enter on the field to keep its current value. Ctrl+C to exit.)"),i.blank(),r=await y(r,{onlySection:"google-places",banner:`Google rejected the key:
25
- ${p}
27
+ paste it below. Press Enter to keep the current value.`}),await C(n,r)}i.blank();for(let u=1;;u++){let f=g.spinner();f.start("Validating Google Places API key");let d=await z(r.googlePlacesApiKey??"");if(d.ok){f.stop("Google Places API key accepted");break}f.stop(d.message??"Google Places check failed",2),u>=A&&L("Google Places",d.message??"Google Places check failed");let y=d.message??"Google Places check failed";i.blank(),i.warn(`Bouncing you back into the Google Places section so you can fix it in place. (attempt ${u+1} of ${A})`),i.detail("(Press Enter on the field to keep its current value. Ctrl+C to exit.)"),i.blank(),r=await b(r,{onlySection:"google-places",banner:`Google rejected the key:
28
+ ${y}
26
29
 
27
30
  Generate (or fix) the key at https://console.cloud.google.com
28
- and paste it below. Press Enter to keep the current value.`}),await S(n,r)}if(r.outreachEnabled&&r.outreachChannels?.includes("email")&&r.emailAddress&&r.emailPassword&&r.emailSmtpHost&&r.emailSmtpPort&&r.emailImapHost&&r.emailImapPort)for(i.blank();;){let u=d.spinner();u.start("Validating email credentials (SMTP + IMAP)");let o=await le({emailAddress:r.emailAddress,emailPassword:r.emailPassword,emailSmtpHost:r.emailSmtpHost,emailSmtpPort:r.emailSmtpPort,emailImapHost:r.emailImapHost,emailImapPort:r.emailImapPort});if(o.ok){u.stop("Email credentials accepted (SMTP + IMAP)");break}u.stop(o.message??"Email check failed",2);let p=o.message??"Email check failed";i.blank(),i.warn("Bouncing you back into the email section so you can fix it in place."),i.detail("(Press Enter on any field to keep its current value. Ctrl+C to exit.)"),i.blank(),r=await y(r,{onlySection:"email",banner:`${o.failedAt==="imap"?"IMAP":"SMTP"} check failed:
29
- ${p}
31
+ and paste it below. Press Enter to keep the current value.`}),await C(n,r)}if(r.outreachEnabled&&r.outreachChannels?.includes("email")&&r.emailAddress&&r.emailPassword&&r.emailSmtpHost&&r.emailSmtpPort&&r.emailImapHost&&r.emailImapPort){i.blank();for(let u=1;;u++){let f=g.spinner();f.start("Validating email credentials (SMTP + IMAP)");let d=await ve({emailAddress:r.emailAddress,emailPassword:r.emailPassword,emailSmtpHost:r.emailSmtpHost,emailSmtpPort:r.emailSmtpPort,emailImapHost:r.emailImapHost,emailImapPort:r.emailImapPort});if(d.ok){f.stop("Email credentials accepted (SMTP + IMAP)");break}f.stop(d.message??"Email check failed",2),u>=A&&L("Email",d.message??"Email check failed");let y=d.message??"Email check failed";i.blank(),i.warn(`Bouncing you back into the email section so you can fix it in place. (attempt ${u+1} of ${A})`),i.detail("(Press Enter on any field to keep its current value. Ctrl+C to exit.)"),i.blank(),r=await b(r,{onlySection:"email",banner:`${d.failedAt==="imap"?"IMAP":"SMTP"} check failed:
32
+ ${y}
30
33
 
31
34
  Re-enter the email address, app password, or host details below.
32
- Press Enter on any field to keep its current value.`}),await S(n,r)}if(r.outreachEnabled&&r.outreachChannels?.includes("sms")&&r.smsProvider==="twilio"&&r.twilioAccountSid&&r.twilioAuthToken)for(i.blank();;){let u=d.spinner();u.start("Validating Twilio credentials");let o=await ce({accountSid:r.twilioAccountSid,authToken:r.twilioAuthToken});if(o.ok){u.stop(o.friendlyName?`Twilio credentials accepted (${o.friendlyName})`:"Twilio credentials accepted");break}u.stop(o.message??"Twilio check failed",2);let p=o.message??"Twilio check failed";i.blank(),i.warn("Bouncing you back into the SMS section so you can fix it in place."),i.detail("(Press Enter on any field to keep its current value. Ctrl+C to exit.)"),i.blank(),r=await y(r,{onlySection:"sms",banner:`Twilio rejected the credentials:
33
- ${p}
35
+ Press Enter on any field to keep its current value.`}),await C(n,r)}}if(r.outreachEnabled&&r.outreachChannels?.includes("sms")&&r.smsProvider==="twilio"&&r.twilioAccountSid&&r.twilioAuthToken){i.blank();for(let u=1;;u++){let f=g.spinner();f.start("Validating Twilio credentials");let d=await Ce({accountSid:r.twilioAccountSid,authToken:r.twilioAuthToken});if(d.ok){f.stop(d.friendlyName?`Twilio credentials accepted (${d.friendlyName})`:"Twilio credentials accepted");break}f.stop(d.message??"Twilio check failed",2),u>=A&&L("Twilio",d.message??"Twilio check failed");let y=d.message??"Twilio check failed";i.blank(),i.warn(`Bouncing you back into the SMS section so you can fix it in place. (attempt ${u+1} of ${A})`),i.detail("(Press Enter on any field to keep its current value. Ctrl+C to exit.)"),i.blank(),r=await b(r,{onlySection:"sms",banner:`Twilio rejected the credentials:
36
+ ${y}
34
37
 
35
38
  Re-enter the Account SID or Auth Token below.
36
- Press Enter on any field to keep its current value.`}),await S(n,r)}if(r.outreachEnabled&&r.outreachChannels?.includes("sms")&&r.smsProvider==="imessage"&&r.testPhone)for(i.blank();;){let u=d.spinner();u.start("Validating iMessage setup (sending a test ping)");let o=await ue({testPhone:r.testPhone});if(o.ok){u.stop("iMessage dispatched (check your phone for the test ping)");break}u.stop(o.message??"iMessage check failed",2);let p=o.message??"iMessage check failed";i.blank(),i.warn("Bouncing you back into the SMS section so you can fix it in place."),i.detail("(Press Enter on any field to keep its current value. Ctrl+C to exit.)"),i.blank(),r=await y(r,{onlySection:"sms",banner:`iMessage check failed:
37
- ${p}
39
+ Press Enter on any field to keep its current value.`}),await C(n,r)}}if(r.outreachEnabled&&r.outreachChannels?.includes("sms")&&r.smsProvider==="imessage"&&r.testPhone){i.blank();for(let u=1;;u++){let f=g.spinner();f.start("Validating iMessage setup (sending a test ping)");let d=await Oe({testPhone:r.testPhone});if(d.ok){f.stop("iMessage dispatched (check your phone for the test ping)");break}f.stop(d.message??"iMessage check failed",2),u>=A&&L("iMessage",d.message??"iMessage check failed");let y=d.message??"iMessage check failed";i.blank(),i.warn(`Bouncing you back into the SMS section so you can fix it in place. (attempt ${u+1} of ${A})`),i.detail("(Press Enter on any field to keep its current value. Ctrl+C to exit.)"),i.blank(),r=await b(r,{onlySection:"sms",banner:`iMessage check failed:
40
+ ${y}
38
41
 
39
42
  Either fix the underlying issue (sign in to Messages.app, configure
40
43
  Text Message Forwarding) and press Enter through, or change SMS
41
- provider to Twilio.`}),await S(n,r)}if(r.telegramBotToken&&r.telegramChatId)for(i.blank();;){let u=d.spinner();u.start("Validating Telegram bot + chat (sending a test ping)");let o=await G({botToken:r.telegramBotToken,chatId:r.telegramChatId});if(o.ok){u.stop(o.botHandle?`Telegram bot ${o.botHandle} accepted, test message delivered`:"Telegram credentials accepted, test message delivered");break}u.stop(o.message??"Telegram check failed",2);let p=o.message??"Telegram check failed";i.blank(),i.warn("Bouncing you back into the Telegram section so you can fix it in place."),i.detail("(Press Enter on any field to keep its current value. Ctrl+C to exit.)"),i.blank(),r=await y(r,{onlySection:"telegram",banner:`Telegram check failed:
42
- ${p}
44
+ provider to Twilio.`}),await C(n,r)}}if(r.telegramBotToken&&r.telegramChatId){i.blank();let u=0,f=!1;for(;!f;){u++;let d=g.spinner();d.start("Validating Telegram bot + chat (sending a test ping)");let y=await fe({botToken:r.telegramBotToken,chatId:r.telegramChatId});if(y.ok){d.stop(y.botHandle?`Telegram bot ${y.botHandle} accepted, test message delivered`:"Telegram credentials accepted, test message delivered"),f=!0;break}d.stop(y.message??"Telegram check failed",2);let Qe=y.message??"Telegram check failed";i.blank(),i.warn("Telegram is optional \u2014 you can skip it now and wire it up later by editing `.env`."),i.blank();let se=await g.select({message:"What would you like to do?",initialValue:u>=2?"skip":"retry",options:[{value:"retry",label:"Retry the check (good if it was a momentary network blip)"},{value:"edit",label:"Edit my bot token / chat ID and retry"},{value:"skip",label:"Skip Telegram for now (I'll add it later by editing .env)"}]});if(g.isCancel(se)&&(g.cancel("Setup cancelled. Your .env is saved \u2014 re-run `klaudius init` to resume."),process.exit(0)),se==="skip"){r.telegramBotToken=void 0,r.telegramChatId=void 0,await C(n,r),i.success("Skipped Telegram. Setup will continue."),i.detail("Wire it up later by pasting your bot token and chat ID into the TELEGRAM_BOT_TOKEN and TELEGRAM_CHAT_ID lines of `.env`."),f=!0;break}if(se==="edit"&&(i.blank(),r=await b(r,{onlySection:"telegram",banner:`Telegram check failed:
45
+ ${Qe}
43
46
 
44
47
  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.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
- "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.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);
48
+ Press Enter on any field to keep its current value.`}),await C(n,r),!r.telegramBotToken||!r.telegramChatId)){i.success("Skipped Telegram. Setup will continue."),f=!0;break}}}i.blank();let k=g.spinner();k.start("Downloading Klaudius template");let P;try{P=(await G({licenseKey:o,machineId:H(),hostname:process.platform==="win32"?process.env.COMPUTERNAME:void 0,os:`${process.platform}-${process.arch}`,destDir:n})).filesExtracted,k.stop(`Downloaded ${P} files into ${m.bold(n)}`)}catch(u){k.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 je(Z(n,"clients"),{recursive:!0});let p=await he(n);await Y(n,Se("0.1.0",p)),i.success("Wrote .klaudius/manifest.json");let E=!1,R="";if(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=g.spinner();u.start("Applying schema to Supabase");let f=await Ie({projectRoot:n,pat:r.supabasePat,projectRef:r.supabaseProjectRef});f.ok?u.stop("Schema applied (or already present)"):(u.stop(f.message??"Schema apply failed",2),E=!0,R=`https://supabase.com/dashboard/project/${r.supabaseProjectRef}/sql/new`)}let _=[],V={};if(e.skipInstall)i.warn("--skip-install set; you'll need to run npm install + pip install + playwright/patchright install yourself.");else{i.blank(),i.heading("Installing dependencies");let u=await Le(n);i.detail(`npm: ${u.npm}, pip: ${u.pip}, playwright: ${u.playwright}, patchright: ${u.patchright}`),u.npm==="failed"&&_.push("npm"),u.pip==="failed"&&_.push("pip"),u.playwright==="failed"&&_.push("playwright"),u.patchright==="failed"&&_.push("patchright"),V=u.hints??{}}i.blank();let de=[..._,...E?["schema"]:[]];if(de.length>0){g.outro(m.yellow(m.bold(`Setup paused \u2014 ${de.join(", ")} failed.`))),console.log(m.dim("Your credentials and project files are all saved.")),i.blank();for(let[d,y]of Object.entries(V))console.log(m.yellow(`Likely cause of ${d} failure:`)),console.log(m.dim(` ${y}`)),i.blank();console.log(m.dim("To finish, once you've addressed any causes above:")),i.blank(),n!==process.cwd()&&console.log(` ${m.cyan(`cd ${e.target}`)}`),_.length>0&&console.log(` ${m.cyan("bash setup.sh")} ${m.dim("# retry the failing installers")}`),E&&(console.log(m.dim(" Apply scripts/schema.sql in the Supabase SQL editor:")),console.log(` ${m.cyan(R)}`),console.log(m.dim(" Open scripts/schema.sql, copy the contents, paste into the editor, click Run."))),i.blank(),_.includes("pip")&&!V.pip&&(console.log(m.dim("If pip fails again with a build error, check `python3 --version` \u2014 Klaudius needs")),console.log(m.dim("Python 3.10+. Install a newer one from https://www.python.org/downloads/macos/")),console.log(m.dim("and open a fresh Terminal before retrying.")),i.blank());let f=["npm","playwright","patchright"].filter(d=>_.includes(d)&&!V[d]);if(f.length>0){let d=f.length===1?"keeps":"keep";console.log(m.dim(`If ${f.join(" / ")} ${d} failing with network errors (ECONNRESET,`)),console.log(m.dim("ETIMEDOUT, ENOTFOUND), they're transient \u2014 try a different network or your")),console.log(m.dim("phone's hotspot, then re-run `bash setup.sh`.")),i.blank()}console.log(m.dim("For a structured diagnostic of every check (Python / Node / installers / files)")),console.log(m.dim(`run ${m.cyan("klaudius doctor")} in this folder.`)),i.blank(),console.log(m.dim("Stuck? Email hello@klaudius.dev with the last ~20 lines of the install output.")),i.blank();return}g.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 C(e,s){await je(e,{recursive:!0}),await Ke(Z(e,".env"),B(s),"utf-8"),await Ke(Z(e,".mcp.json"),F(s),"utf-8"),await Q(e,"supabase")}async function Ht(e){try{(await Dt(e)).isDirectory()||(i.error(`Target exists but is not a directory: ${e}`),process.exit(1))}catch{return{kind:"fresh"}}let s=await Ut(e),n=s.filter(c=>!c.startsWith(".DS_Store"));if(n.length===0)return{kind:"fresh"};let t=new Set([".env",".mcp.json",".claude"]),o=n.every(c=>t.has(c)),a=n.includes(".env"),l=s.includes(".klaudius");return o&&a&&!l?{kind:"partial",envPath:Z(e,".env")}:{kind:"occupied",entries:n}}async function Vt(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 g.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 g.isCancel(s)&&(g.cancel("Setup cancelled. Existing .env left in place."),process.exit(0)),s}async function Bt(e){if(e&&e.trim().length>0)return e;g.intro("Klaudius");let s=await g.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 g.isCancel(s)&&(g.cancel("Setup cancelled. No files written."),process.exit(0)),s}import{readFile as Ft,writeFile as Ve,access as Gt}from"fs/promises";import{join as ae}from"path";import*as ee from"@clack/prompts";import Yt from"picocolors";async function Fe(e){let s=e.projectRoot??process.cwd(),n=ae(s,".env"),t=ae(s,".mcp.json"),o=await Be(ae(s,".klaudius","manifest.json")),a=await Be(n);!o&&!a&&(i.error("This doesn't look like a Klaudius project."),i.detail("Expected `.klaudius/manifest.json` or `.env` in this folder. Run `npx klaudius init` first."),process.exit(1));let l=X();l.ok||(i.error(l.message??"Node.js version check failed."),i.blank(),i.detail("Once you've installed Node.js 22.15+, re-run `klaudius configure` in this folder."),process.exit(1));let c={};try{let w=await Ft(n,"utf-8");c=v(w)}catch{i.warn("No .env found. Starting from an empty configuration.")}let h=ke(c);ee.intro("Reconfigure Klaudius"),i.detail("Press Enter on any prompt to keep its current value."),i.blank();let r=await b(h);if(r.supabaseUrl!==c.SUPABASE_URL||r.supabaseServiceKey!==c.SUPABASE_SERVICE_KEY){i.heading("Re-validating Supabase credentials");let w=await $({url:r.supabaseUrl,serviceKey:r.supabaseServiceKey});w.ok||(i.error(w.message??"Supabase check failed"),i.warn("Aborting. .env not modified."),process.exit(1)),i.success("Supabase credentials accepted")}await Ve(n,B(r),"utf-8"),i.success("Updated .env"),await Ve(t,F(r),"utf-8"),i.success("Updated .mcp.json"),await Q(s,"supabase"),i.blank(),ee.outro(Yt.green("Configuration updated."))}async function Be(e){try{return await Gt(e),!0}catch{return!1}}import{copyFile as Ge,mkdir as Ye,readdir as Wt,readFile as zt,rm as te,writeFile as We,access as qt}from"fs/promises";import{dirname as ze,join as I,relative as Jt}from"path";import{tmpdir as Xt}from"os";import*as D from"@clack/prompts";import le from"picocolors";var ce="0.7.0";async function qe(e){let s=e.projectRoot??process.cwd(),n=await Pe(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,o=I(s,".env");try{let p=await zt(o,"utf-8"),E=v(p);t=E.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 R=ge(p,E,[{key:"PRICING_TERMS",value:"one-off, no monthly fees",comment:"Phrasing slotted next to ${PRICING} in outreach copy. See template/CLAUDE.md."},{key:"VERCEL_SCOPE",value:"",comment:"Vercel team slug. Leave blank unless `vercel link` errors about scope (post-2024 multi-team Vercel accounts)."}]);R!==null&&(await We(o,R,"utf-8"),i.detail("Backfilled new env keys"))}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)}D.intro(`Update from ${n.templateVersion} \u2192 ${ce}`);let a=I(Xt(),`klaudius-update-${process.pid}-${Date.now()}`),l;try{i.heading("Downloading the latest Klaudius template");let p=await G({licenseKey:t,machineId:H(),hostname:process.env.HOSTNAME??process.env.COMPUTERNAME,os:`${process.platform}-${process.arch}`,destDir:a});i.success(`Fetched ${p.filesExtracted} canonical files`),l=a}catch(p){i.error(p.message),i.detail("Update aborted. Your project is unchanged."),await te(a,{recursive:!0,force:!0}).catch(()=>{}),process.exit(1)}let c=await Qt(l,s,n),h=c.filter(p=>p.state==="new"),r=c.filter(p=>p.state==="clean-update"),w=c.filter(p=>p.state==="user-only"),k=c.filter(p=>p.state==="conflict"),P=c.filter(p=>p.state==="no-change");if(i.blank(),i.heading("Change summary"),i.detail(`${U(P.length,"file")} unchanged`),i.detail(`${U(h.length,"new file")}`),i.detail(`${U(r.length,"file")} updated cleanly (no local edits)`),i.detail(`${U(w.length,"file")} you've edited (no upstream change \u2014 left alone)`),i.detail(`${U(k.length,"file")} with conflicts (you edited AND we have a new version)`),r.length===0&&h.length===0&&k.length===0){i.blank(),D.outro(le.green("Already up to date.")),await te(a,{recursive:!0,force:!0}).catch(()=>{});return}i.blank(),i.heading("Applying clean updates");for(let p of[...h,...r]){let E=I(l,p.path),R=I(s,p.path);await Ye(ze(R),{recursive:!0}),await Ge(E,R),n.files[p.path]=p.canonicalHash}if(i.success(`Wrote ${U(h.length+r.length,"file")}`),k.length>0){i.blank(),i.heading(`${k.length} conflict(s) need resolution`);let p=I(s,".klaudius","incoming");try{await te(p,{recursive:!0,force:!0})}catch{}for(let E of k){let R=I(l,E.path),_=I(p,E.path);await Ye(ze(_),{recursive:!0}),await Ge(R,_)}await We(I(s,".klaudius","conflicts.md"),Zt(k,n.templateVersion,ce),"utf-8"),i.detail("Staged canonical versions to .klaudius/incoming/"),i.detail("Wrote conflict report to .klaudius/conflicts.md")}n.updatedAt=new Date().toISOString(),k.length===0&&(n.templateVersion=ce),await Y(s,n),i.blank(),k.length===0?D.outro(le.green("Update complete.")):D.outro(le.yellow(`${k.length} conflict(s) staged. Open Claude Code in this project and ask:
49
+ "Run /resolve-conflicts"`)),await te(a,{recursive:!0,force:!0}).catch(()=>{})}async function Qt(e,s,n){let t=await Je(e),o=[];for(let a of t){let l=await ne(I(e,a)),c=n.files[a],h=I(s,a),r;try{await qt(h),r=await ne(h)}catch{o.push({path:a,state:"new",canonicalHash:l,manifestHash:c});continue}l===c?r===c?o.push({path:a,state:"no-change",canonicalHash:l,localHash:r,manifestHash:c}):o.push({path:a,state:"user-only",canonicalHash:l,localHash:r,manifestHash:c}):r===c?o.push({path:a,state:"clean-update",canonicalHash:l,localHash:r,manifestHash:c}):o.push({path:a,state:"conflict",canonicalHash:l,localHash:r,manifestHash:c})}return o}function U(e,s){return`${e} ${s}${e===1?"":"s"}`}async function Je(e,s=e){let n=await Wt(e,{withFileTypes:!0}),t=[];for(let o of n){let a=I(e,o.name);o.isDirectory()?t.push(...await Je(a,s)):o.isFile()&&t.push(Jt(s,a))}return t}function Zt(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 o of e)t.push(`## ${o.path}`),t.push(""),t.push(`- Your version (with your local edits): \`${o.path}\``),t.push(`- New canonical version: \`.klaudius/incoming/${o.path}\``),t.push(`- Hash at last install/update: \`${(o.manifestHash??"(none)").slice(0,16)}\``),t.push(`- Your local hash now: \`${(o.localHash??"(missing)").slice(0,16)}\``),t.push(`- New canonical hash: \`${o.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(`
50
+ `)}import{access as es,readFile as ts,stat as ss}from"fs/promises";import{join as N}from"path";import S from"picocolors";import{spawn as ns}from"child_process";async function Xe(e){let s=e.projectRoot??process.cwd(),n=[];console.log(S.bold(`Klaudius doctor \u2014 checking your project\u2026
51
+ `)),n.push(await is()),n.push(await os());let t=N(s,".env"),o=N(s,".klaudius","manifest.json"),a=N(s,".mcp.json"),l=N(s,"setup.sh"),c=await K(t),h=await K(o),r=await K(a),w=await K(l);if(n.push({name:".env present",status:c?"ok":"fail",detail:c?t:"missing",nextAction:c?void 0:"Run `klaudius init` in this folder."}),n.push({name:"Template scaffolded (.klaudius/manifest.json)",status:h?"ok":"fail",detail:h?o:"missing",nextAction:h?void 0:c?"Init didn't finish \u2014 run `klaudius init` and pick Resume.":"Run `klaudius init`."}),n.push({name:".mcp.json present",status:r?"ok":"warn",detail:r?a:"missing",nextAction:r?void 0:"Run `klaudius configure` to regenerate."}),n.push({name:"setup.sh present",status:w?"ok":"warn",detail:w?l:"missing \u2014 re-run init"}),n.push(await rs(s)),n.push(await as(s)),n.push(await ls()),!c)n.push({name:"Credential validations",status:"skip",detail:".env missing"});else if(e.offline)n.push({name:"Credential validations",status:"skip",detail:"--offline set"});else{let p=await ts(t,"utf-8"),E=v(p);await cs(E,n)}console.log();for(let p of n){let E=p.status==="ok"?S.green("\u2713"):p.status==="warn"?S.yellow("\u26A0"):p.status==="skip"?S.dim("\xB7"):S.red("\u2717");console.log(` ${E} ${p.name}${p.detail?S.dim(` \u2014 ${p.detail}`):""}`),p.nextAction&&console.log(` ${S.dim("\u2192")} ${S.cyan(p.nextAction)}`)}let k=n.filter(p=>p.status==="fail").length,P=n.filter(p=>p.status==="warn").length;console.log(),k===0&&P===0?console.log(S.green(S.bold("Everything checks out. You're good to go."))):k===0?console.log(S.yellow(S.bold(`${P} warning${P===1?"":"s"} \u2014 review above.`))):(console.log(S.red(S.bold(`${k} failed check${k===1?"":"s"} \u2014 see the \u2192 next actions above.`))),process.exitCode=1),console.log()}async function K(e){try{return await es(e),!0}catch{return!1}}async function is(){let e=await J();return e.ok?{name:"Python \u2265 3.10",status:"ok",detail:e.version}:{name:"Python \u2265 3.10",status:"fail",detail:e.version??"not found",nextAction:"Install Python 3.10+ from https://www.python.org/downloads/"}}async function os(){let e=process.version,s=/^v(\d+)\.(\d+)\.(\d+)$/.exec(e);if(!s)return{name:"Node \u2265 22.15",status:"warn",detail:e};let n=parseInt(s[1],10),t=parseInt(s[2],10);return n<22||n===22&&t<15?{name:"Node \u2265 22.15",status:"fail",detail:e,nextAction:"Upgrade Node.js to 22.15+ from https://nodejs.org/"}:{name:"Node \u2265 22.15",status:"ok",detail:e}}async function rs(e){let s=N(e,"node_modules");if(!await K(s))return{name:"node_modules installed",status:"fail",detail:"missing",nextAction:"Run `bash setup.sh` (or `npm install`) in this folder."};try{if(!(await ss(s)).isDirectory())return{name:"node_modules installed",status:"warn",detail:"exists but isn't a directory"}}catch{}return{name:"node_modules installed",status:"ok"}}async function as(e){let s=process.platform==="darwin"?N(process.env.HOME??"","Library","Caches","ms-playwright"):process.platform==="win32"?N(process.env.USERPROFILE??"","AppData","Local","ms-playwright"):N(process.env.HOME??"",".cache","ms-playwright");return await K(s)?{name:"Playwright browser installed",status:"ok",detail:s}:{name:"Playwright browser installed",status:"fail",detail:"no browser cache found",nextAction:"Run `npx playwright install chromium` in this folder."}}async function ls(){let e=["supabase","dotenv","requests","httpx","PIL"],s=`import sys
52
+ try:
53
+ ${e.map(t=>` import ${t}`).join(`
54
+ `)}
55
+ except Exception as e:
56
+ print(str(e))
57
+ sys.exit(1)
58
+ `,n=await ds("python3",["-c",s]);return n.code===0?{name:"Python packages installed",status:"ok",detail:e.join(", ")}:{name:"Python packages installed",status:"fail",detail:(n.stdout||n.stderr).trim().split(`
59
+ `).pop()??"import failed",nextAction:"Run `bash setup.sh` in this folder."}}async function cs(e,s){if(e.SUPABASE_URL&&e.SUPABASE_SERVICE_KEY){let n=await $({url:e.SUPABASE_URL,serviceKey:e.SUPABASE_SERVICE_KEY});s.push({name:"Supabase credentials",status:n.ok?"ok":"fail",detail:n.ok?"connected":n.message,nextAction:n.ok?void 0:"Run `klaudius configure` to fix the URL or key."})}else s.push({name:"Supabase credentials",status:"fail",detail:"missing in .env"});if(e.VERCEL_TOKEN){let n=await W(e.VERCEL_TOKEN);s.push({name:"Vercel token",status:n.ok?"ok":"fail",detail:n.ok?n.username??"valid":n.message,nextAction:n.ok?void 0:"Generate a new token at https://vercel.com/account/tokens, then `klaudius configure`."})}else s.push({name:"Vercel token",status:"fail",detail:"missing in .env"});if(e.GOOGLE_PLACES_API_KEY){let n=await z(e.GOOGLE_PLACES_API_KEY);s.push({name:"Google Places API key",status:n.ok?"ok":"fail",detail:n.ok?"accepted":n.message,nextAction:n.ok?void 0:"Fix the key in https://console.cloud.google.com and `klaudius configure`."})}else s.push({name:"Google Places API key",status:"warn",detail:"not set (find skill will fail until you add one)"});if(e.TELEGRAM_BOT_TOKEN&&e.TELEGRAM_CHAT_ID){let n=await me({botToken:e.TELEGRAM_BOT_TOKEN});s.push({name:"Telegram bot token",status:n.ok?"ok":"fail",detail:n.ok?`bot ${n.botHandle??"ok"} (chat_id not tested)`:n.message,nextAction:n.ok?void 0:"Fix TELEGRAM_BOT_TOKEN in .env (or blank both Telegram lines to disable)."})}else e.TELEGRAM_BOT_TOKEN||e.TELEGRAM_CHAT_ID?s.push({name:"Telegram credentials",status:"warn",detail:"partial config (one of bot token / chat ID set, other missing)",nextAction:"Either set both, or blank both to disable Telegram notifications."}):s.push({name:"Telegram credentials",status:"skip",detail:"not configured (optional)"})}function ds(e,s){return new Promise(n=>{let t=ns(e,s,{stdio:["ignore","pipe","pipe"],shell:!1}),o="",a="";t.stdout?.on("data",l=>{o+=l.toString()}),t.stderr?.on("data",l=>{a+=l.toString()}),t.on("close",l=>n({code:l??1,stdout:o,stderr:a})),t.on("error",()=>n({code:1,stdout:o,stderr:a}))})}var ps="0.7.0",j=new us;j.name("klaudius").description("Scaffold and maintain a Klaudius pipeline project").version(ps);j.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 He({target:e,license:s.license,skipInstall:!!s.skipInstall,skipSchema:!!s.skipSchema})}catch(n){console.error("init failed:",n.message),process.exit(1)}});j.command("configure").description("Re-run the setup wizard with current values pre-filled").action(async()=>{try{await Fe({})}catch(e){console.error("configure failed:",e.message),process.exit(1)}});j.command("update").description("Pull latest template; produce a conflicts report for any files you've modified").action(async()=>{try{await qe({})}catch(e){console.error("update failed:",e.message),process.exit(1)}});j.command("doctor").description("Diagnose a broken install \u2014 env versions, files, installed deps, credentials").option("--offline","Skip credential validations that require network calls").action(async e=>{try{await Xe({offline:!!e.offline})}catch(s){console.error("doctor failed:",s.message),process.exit(1)}});j.parse(process.argv);
@@ -0,0 +1,6 @@
1
+ import{appendFileSync as d}from"fs";import{tmpdir as g}from"os";import{join as m}from"path";var c=m(g(),"klaudius-wizard-debug.log"),h=process.env.KLAUDIUS_WIZARD_DEBUG==="1",l=!1;function f(o,e){if(h)try{l||(l=!0,d(c,`
2
+ === klaudius wizard debug session \u2014 ${new Date().toISOString()} ===
3
+ node ${process.version}, platform ${process.platform}, tty=${process.stdin.isTTY}
4
+ `));let t=`${Date.now()} `,s=e===void 0?"":" "+JSON.stringify(e,null,0).slice(0,400);d(c,t+o+s+`
5
+ `)}catch{}}var y=c,u=!1;function w(){if(u)return;u=!0;let o=process.exit.bind(process);process.exit=(e=>{let t=new Error("process.exit called").stack??"(no stack)";return f("process.exit called",{code:e,stack:t.split(`
6
+ `).slice(0,8).join(" | ")}),o(e)})}function i(o){if(!(o instanceof Error))return String(o);let e=o.cause;if(!e||!(e instanceof Error))return o.message;let t=e.code,s=e.message;return t&&s&&s!==t?`${o.message} (${t}: ${s})`:t?`${o.message} (${t})`:s?`${o.message} (${s})`:o.message}async function I(o){let e;try{let n=await fetch(`https://api.telegram.org/bot${encodeURIComponent(o.botToken)}/getUpdates`);if(e=await n.json(),!e.ok){let a=e.description??`HTTP ${n.status}`;return/webhook/i.test(a)?{ok:!1,reason:"webhook_set",message:"This bot has a webhook set, so getUpdates is unavailable. Auto-detect can't run; you'll need to paste the chat ID manually."}:{ok:!1,reason:"bad_token",message:`Telegram rejected the bot token: ${a}. Double-check the token from @BotFather.`}}}catch(n){return{ok:!1,reason:"network",message:`Could not reach Telegram: ${i(n)}`}}let t=e.result??[];if(t.length===0)return{ok:!1,reason:"no_messages",message:"No messages yet. Send your bot a regular text message in Telegram, then retry."};let s=t[t.length-1],r=s?.message?.chat?.id??s?.edited_message?.chat?.id??s?.channel_post?.chat?.id;return r===void 0?{ok:!1,reason:"no_messages",message:"Got an update from Telegram but couldn't find a chat ID in it. Send another text message and retry."}:{ok:!0,chatId:String(r)}}async function p(o){try{let e=await fetch(`https://api.telegram.org/bot${encodeURIComponent(o.botToken)}/getMe`),t=await e.json();return t.ok?{ok:!0,botHandle:t.result?.username?`@${t.result.username}`:t.result?.first_name}:{ok:!1,failedAt:"token",message:`Telegram rejected the bot token: ${t.description??`HTTP ${e.status}`}. Generate a new token by messaging @BotFather \u2192 /mytoken or /newbot.`}}catch(e){return{ok:!1,failedAt:"token",message:`Could not reach Telegram: ${i(e)}`}}}async function A(o){let e=await p({botToken:o.botToken});if(!e.ok)return e;let t=e.botHandle;try{let s=await fetch(`https://api.telegram.org/bot${encodeURIComponent(o.botToken)}/sendMessage`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({chat_id:o.chatId,text:"\u2713 Klaudius connected. You'll get a notification here whenever outreach lands or a candidate replies. (You can ignore this setup message.)",disable_notification:!0})}),r=await s.json();if(!r.ok){let n=r.description??`HTTP ${s.status}`,a="";return/chat not found/i.test(n)?a=" The chat_id is wrong, or the bot hasn't been added to that chat yet. If it's a group, you must add the bot to the group first; then send /start to it.":(/bot was blocked/i.test(n)||/forbidden/i.test(n))&&(a=" The bot has been blocked by the chat, or never received a /start. Open Telegram, find your bot, and send /start to wake it up."),{ok:!1,failedAt:"chat",botHandle:t,message:`Telegram delivered the bot token check but couldn't send to chat_id ${o.chatId}: ${n}.${a}`}}}catch(s){return{ok:!1,failedAt:"chat",botHandle:t,message:`Could not reach Telegram for chat_id check: ${i(s)}`}}return{ok:!0,botHandle:t}}export{f as a,y as b,w as c,i as d,I as e,p as f,A as g};
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "klaudius",
3
- "version": "0.5.1",
3
+ "version": "0.7.0",
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": {
@@ -29,7 +29,7 @@
29
29
  "prepublishOnly": "npm run build"
30
30
  },
31
31
  "engines": {
32
- "node": ">=22"
32
+ "node": ">=22.15"
33
33
  },
34
34
  "dependencies": {
35
35
  "@clack/prompts": "^0.7.0",
@@ -1,6 +0,0 @@
1
- import{appendFileSync as c}from"fs";import{tmpdir as g}from"os";import{join as u}from"path";var r=u(g(),"klaudius-wizard-debug.log"),h=process.env.KLAUDIUS_WIZARD_DEBUG==="1",d=!1;function m(o,t){if(h)try{d||(d=!0,c(r,`
2
- === klaudius wizard debug session \u2014 ${new Date().toISOString()} ===
3
- node ${process.version}, platform ${process.platform}, tty=${process.stdin.isTTY}
4
- `));let e=`${Date.now()} `,s=t===void 0?"":" "+JSON.stringify(t,null,0).slice(0,400);c(r,e+o+s+`
5
- `)}catch{}}var b=r,l=!1;function T(){if(l)return;l=!0;let o=process.exit.bind(process);process.exit=(t=>{let e=new Error("process.exit called").stack??"(no stack)";return m("process.exit called",{code:t,stack:e.split(`
6
- `).slice(0,8).join(" | ")}),o(t)})}async function w(o){let t;try{let a=await fetch(`https://api.telegram.org/bot${encodeURIComponent(o.botToken)}/getUpdates`);if(t=await a.json(),!t.ok){let i=t.description??`HTTP ${a.status}`;return/webhook/i.test(i)?{ok:!1,reason:"webhook_set",message:"This bot has a webhook set, so getUpdates is unavailable. Auto-detect can't run; you'll need to paste the chat ID manually."}:{ok:!1,reason:"bad_token",message:`Telegram rejected the bot token: ${i}. Double-check the token from @BotFather.`}}}catch(a){return{ok:!1,reason:"network",message:`Could not reach Telegram: ${a.message}`}}let e=t.result??[];if(e.length===0)return{ok:!1,reason:"no_messages",message:"No messages yet. Send your bot a regular text message in Telegram, then retry."};let s=e[e.length-1],n=s.message?.chat?.id??s.edited_message?.chat?.id??s.channel_post?.chat?.id;return n===void 0?{ok:!1,reason:"no_messages",message:"Got an update from Telegram but couldn't find a chat ID in it. Send another text message and retry."}:{ok:!0,chatId:String(n)}}async function _(o){let t;try{let e=await fetch(`https://api.telegram.org/bot${encodeURIComponent(o.botToken)}/getMe`),s=await e.json();if(!s.ok)return{ok:!1,failedAt:"token",message:`Telegram rejected the bot token: ${s.description??`HTTP ${e.status}`}. Generate a new token by messaging @BotFather \u2192 /mytoken or /newbot.`};t=s.result?.username?`@${s.result.username}`:s.result?.first_name}catch(e){return{ok:!1,failedAt:"token",message:`Could not reach Telegram: ${e.message}`}}try{let e=await fetch(`https://api.telegram.org/bot${encodeURIComponent(o.botToken)}/sendMessage`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({chat_id:o.chatId,text:"\u2713 Klaudius connected. You'll get a notification here whenever outreach lands or a candidate replies. (You can ignore this setup message.)",disable_notification:!0})}),s=await e.json();if(!s.ok){let n=s.description??`HTTP ${e.status}`,a="";return/chat not found/i.test(n)?a=" The chat_id is wrong, or the bot hasn't been added to that chat yet. If it's a group, you must add the bot to the group first; then send /start to it.":(/bot was blocked/i.test(n)||/forbidden/i.test(n))&&(a=" The bot has been blocked by the chat, or never received a /start. Open Telegram, find your bot, and send /start to wake it up."),{ok:!1,failedAt:"chat",botHandle:t,message:`Telegram delivered the bot token check but couldn't send to chat_id ${o.chatId}: ${n}.${a}`}}}catch(e){return{ok:!1,failedAt:"chat",botHandle:t,message:`Could not reach Telegram for chat_id check: ${e.message}`}}return{ok:!0,botHandle:t}}export{m as a,b,T as c,w as d,_ as e};