@seoagent-official/seoagent 1.22.1 → 1.24.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/index.js +165 -23
- package/package.json +1 -1
- package/skills/seoagent.md +36 -4
package/index.js
CHANGED
|
@@ -1,31 +1,31 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
-
import{Command as
|
|
2
|
+
import{Command as zr}from"commander";import{intro as uo,outro as po,text as It,select as _e,spinner as fo,isCancel as j,cancel as N,confirm as go}from"@clack/prompts";import{existsSync as Je,mkdirSync as qe,readFileSync as yn,writeFileSync as Xe}from"fs";import{join as me}from"path";var A=[".env.local",".env.production",".env"],Ke=["NEXT_PUBLIC_SITE_URL","SITE_URL","NEXT_PUBLIC_URL","NEXTAUTH_URL","VITE_SITE_URL"];import{homedir as dn}from"os";import{join as z}from"path";var m=".seoagent",ze=["audit","strategy/clusters","briefs","content","content/images","performance"],C={AGENTS:".agents/skills/seoagent",CLAUDE:".claude/skills/seoagent"},Y=".claude/settings.json",Ye=".pull-receipt.json",Ve=`# SEOAgent local files
|
|
3
3
|
.legacy/
|
|
4
4
|
content/images/*
|
|
5
5
|
!content/images/.gitkeep
|
|
6
6
|
.pull-receipt.json
|
|
7
|
-
`,pn=process.env.XDG_CONFIG_HOME&&process.env.XDG_CONFIG_HOME.trim().length>0?process.env.XDG_CONFIG_HOME:z(
|
|
8
|
-
`)}}function
|
|
7
|
+
`,pn=process.env.XDG_CONFIG_HOME&&process.env.XDG_CONFIG_HOME.trim().length>0?process.env.XDG_CONFIG_HOME:z(dn(),".config"),S=z(pn,"seoagent"),v=z(S,"auth.json"),pe=z(S,"state");var D="https://seoagent.com",R=process.env.SEOAGENT_API_BASE&&process.env.SEOAGENT_API_BASE.trim().length>0?process.env.SEOAGENT_API_BASE.replace(/\/$/,""):D,k={BASE:R,PRICING:`${R}/pricing`,LEAD_API:`${R}/api/cli/lead`,CLI_AUTH_PAGE:`${R}/cli/auth`,CLI_AUTH_POLL:`${R}/api/cli/auth/poll`,CLI_SYNC:`${R}/api/cli/sync`};var V="0.2.0";import{existsSync as fn,readFileSync as gn}from"fs";var J="---";function mn(e){let t=e.trim();return t===""?"":t==="null"||t==="~"?null:t==="true"?!0:t==="false"?!1:t.startsWith('"')&&t.endsWith('"')||t.startsWith("'")&&t.endsWith("'")?t.slice(1,-1):/^-?\d+$/.test(t)||/^-?\d+\.\d+$/.test(t)?Number(t):t}function fe(e){let t=e.split(/\r?\n/);if(t[0]?.trim()!==J)return{data:{},body:e};let n={},o=1;for(;o<t.length;o++){if(t[o].trim()===J){o++;break}let r=t[o],i=r.indexOf(":");if(i===-1)continue;let s=r.slice(0,i).trim(),a=r.slice(i+1);s&&(n[s]=mn(a))}return{data:n,body:t.slice(o).join(`
|
|
8
|
+
`)}}function q(e){if(!fn(e))return null;try{return fe(gn(e,"utf-8"))}catch{return null}}function hn(e){if(e===null)return"null";if(typeof e=="boolean")return e?"true":"false";if(typeof e=="number")return String(e);let t=String(e);return t===""||/[:#\[\]{}&*!|>'"%@`,]/.test(t)||/^\s|\s$/.test(t)?`"${t.replace(/"/g,'\\"')}"`:t}function ge(e,t=""){let n=[J];for(let[r,i]of Object.entries(e))i!==void 0&&n.push(`${r}: ${hn(i)}`);n.push(J);let o=t.length===0?"":t.startsWith(`
|
|
9
9
|
`)?t:`
|
|
10
10
|
${t}`;return n.join(`
|
|
11
11
|
`)+o+(o.endsWith(`
|
|
12
12
|
`)?"":`
|
|
13
|
-
`)}var
|
|
14
|
-
`)}function
|
|
13
|
+
`)}var kn=new Set(["strapi","wordpress","sanity","contentful","ghost","webflow","shopify","payload","directus","mdx-local","none"]),Sn="project.md";function M(e){return me(e,m)}function X(e){return me(M(e),Sn)}function Ze(e){return Je(X(e))}function f(e){let t=q(X(e));if(!t)return null;let n=t.data;if(typeof n.domain!="string"||!n.domain)return null;let o=typeof n.image_provider=="string"?n.image_provider:void 0,r=typeof n.cms=="string"?n.cms:void 0,i=r&&kn.has(r)?r:void 0;return{domain:n.domain,site_type:typeof n.site_type=="string"?n.site_type:"unknown",language:typeof n.language=="string"?n.language:"en",initialized_at:typeof n.initialized_at=="string"?n.initialized_at:"",seoagent_version:typeof n.seoagent_version=="string"?n.seoagent_version:"",image_provider:o==="openai"||o==="fal"||o==="replicate"||o==="none"?o:void 0,cms:i,blog_path:typeof n.blog_path=="string"&&n.blog_path?n.blog_path:void 0}}function xn(e){return["",`# SEOAgent Project \u2014 ${e.domain}`,"","This file holds project-level configuration for the SEOAgent skill.","The skill reads it on every session to know which domain it is working on.","",`Initialized ${e.initialized_at} with seoagent ${e.seoagent_version}.`,""].join(`
|
|
14
|
+
`)}function bn(e){let t={domain:e.domain,site_type:e.site_type,language:e.language,initialized_at:e.initialized_at,seoagent_version:e.seoagent_version};return e.image_provider&&(t.image_provider=e.image_provider),e.cms&&(t.cms=e.cms),e.blog_path&&(t.blog_path=e.blog_path),t}function Qe(e,t){let n=M(e);qe(n,{recursive:!0});let o=ge(bn(t),xn(t));Xe(X(e),o,"utf-8")}function et(e,t){let n=X(e);if(!Je(n))return!1;let o=fe(yn(n,"utf-8")),r={...o.data,...t};return Xe(n,ge(r,o.body),"utf-8"),!0}function tt(e){let t=M(e);for(let n of ze)qe(me(t,n),{recursive:!0})}import{existsSync as Z,mkdirSync as Q,writeFileSync as nt,readFileSync as ot,readdirSync as wn,statSync as vn}from"fs";import{join as b,dirname as rt,relative as _n}from"path";import{fileURLToPath as En}from"url";var In=rt(En(import.meta.url));function it(){return b(In,"skills")}function Pn(){let e=b(it(),"seoagent.md");if(Z(e))return ot(e,"utf-8");throw new Error(`Could not find skill file at: ${e}`)}function st(e,t){let n=[];if(!Z(e))return n;for(let o of wn(e,{withFileTypes:!0})){let r=b(e,o.name),i=b(t,o.name);o.isDirectory()?(Q(i,{recursive:!0}),n.push(...st(r,i))):o.isFile()&&(Q(rt(i),{recursive:!0}),nt(i,ot(r)),n.push(i))}return n}function at(e,t){return $n(e,t).skillFile}function $n(e,t){let n=Z(b(e,".agents")),o=b(e,n?C.AGENTS:C.CLAUDE),r=b(o,"SKILL.md");Q(o,{recursive:!0}),nt(r,t??Pn(),"utf-8");let i=b(it(),"references"),s=b(o,"references"),a=[];try{Z(i)&&vn(i).isDirectory()&&(Q(s,{recursive:!0}),a=st(i,s).map(l=>_n(o,l)))}catch{}return{skillFile:r,referenceFiles:a}}import{existsSync as ee,mkdirSync as he,readFileSync as An,writeFileSync as B}from"fs";import{dirname as ct,join as T}from"path";var Cn="npx -y @seoagent-official/seoagent sync --silent";function lt(e){let t=T(e,m);he(t,{recursive:!0});let n=T(t,".gitignore");B(n,Ve,"utf-8");let o=T(t,"content","images",".gitkeep");return he(ct(o),{recursive:!0}),ee(o)||B(o,"","utf-8"),n}function ye(e){if(!ee(e))return{};try{let t=An(e,"utf-8");return JSON.parse(t)}catch{return{}}}function ke(e){return e.command.includes("@seoagent-official/seoagent")&&e.command.includes("sync")}function ut(e){let t=T(e,Y);he(ct(t),{recursive:!0});let n=ye(t);n.hooks=n.hooks??{};let o=n.hooks.PostToolUse=n.hooks.PostToolUse??[],r=o.find(i=>(i.matcher??"").includes("Write"));return r||(r={matcher:"Write|Edit",hooks:[]},o.push(r)),r.hooks.some(ke)?(B(t,JSON.stringify(n,null,2)+`
|
|
15
15
|
`,"utf-8"),{file:t,added:!1}):(r.hooks.push({type:"command",command:Cn}),B(t,JSON.stringify(n,null,2)+`
|
|
16
|
-
`,"utf-8"),{file:t,added:!0})}function
|
|
17
|
-
`,"utf-8"),{file:t,removed:!0}):{file:t,removed:!1}}import{readFileSync as
|
|
18
|
-
`)}var $n="context.md";function On(e,t){let n=Rn(e,".seoagent",$n),o=dt(t);Tn(n,o,"utf-8")}function ft(e,t){On(e,{business:{type:t||""},writingInstructions:[],referenceUrls:[],topicsToAvoid:[],contentTone:null,additionalNotes:null})}import{existsSync as G,readFileSync as mt,readdirSync as jn,statSync as gt}from"fs";import{join as _}from"path";var Nn=[{cms:"strapi",deps:["strapi","@strapi/strapi","@strapi/"],envKeys:["STRAPI_URL","STRAPI_API_URL","NEXT_PUBLIC_STRAPI_URL","STRAPI_API_TOKEN"],fsMarkers:["strapi/","apps/strapi/","packages/strapi/","cms/"]},{cms:"wordpress",deps:["wpapi","wp-graphql","@wordpress/api-fetch","@wordpress/"],envKeys:["WORDPRESS_API_URL","WP_API_URL","NEXT_PUBLIC_WORDPRESS_URL"],fsMarkers:[]},{cms:"sanity",deps:["@sanity/client","next-sanity","sanity"],envKeys:["SANITY_PROJECT_ID","NEXT_PUBLIC_SANITY_PROJECT_ID","SANITY_API_TOKEN"],fsMarkers:["sanity/","studio/","sanity.config.ts","sanity.config.js"]},{cms:"contentful",deps:["contentful","@contentful/rich-text-react-renderer","@contentful/"],envKeys:["CONTENTFUL_SPACE_ID","CONTENTFUL_ACCESS_TOKEN","NEXT_PUBLIC_CONTENTFUL_SPACE_ID"],fsMarkers:[]},{cms:"ghost",deps:["@tryghost/content-api","@tryghost/"],envKeys:["GHOST_URL","GHOST_API_KEY","NEXT_PUBLIC_GHOST_URL"],fsMarkers:[]},{cms:"webflow",deps:["webflow-api"],envKeys:["WEBFLOW_API_TOKEN","WEBFLOW_SITE_ID"],fsMarkers:[]},{cms:"shopify",deps:["@shopify/hydrogen","@shopify/storefront-api-client","@shopify/shopify-api","@shopify/"],envKeys:["SHOPIFY_STOREFRONT_TOKEN","SHOPIFY_STORE_DOMAIN","SHOPIFY_ADMIN_API_TOKEN"],fsMarkers:["shopify.config.ts","shopify.config.js"]},{cms:"payload",deps:["payload","@payloadcms/next","@payloadcms/"],envKeys:["PAYLOAD_SECRET","PAYLOAD_PUBLIC_SERVER_URL"],fsMarkers:["payload.config.ts","payload.config.js"]},{cms:"directus",deps:["@directus/sdk","directus"],envKeys:["DIRECTUS_URL","DIRECTUS_TOKEN"],fsMarkers:[]}];function Ln(e){let t=_(e,"package.json");if(!G(t))return{deps:{},source:"package.json"};try{let n=JSON.parse(mt(t,"utf-8"));return{deps:{...n.dependencies,...n.devDependencies,...n.peerDependencies},source:"package.json"}}catch{return{deps:{},source:"package.json"}}}function Fn(e,t){for(let n of t)if(n.endsWith("/")){if(e.startsWith(n))return!0}else if(e===n)return!0;return!1}function Un(e){let t={};for(let n of C){let o=_(e,n);if(G(o))try{let r=mt(o,"utf-8");for(let i of r.split(/\r?\n/)){let s=i.trim();if(!s||s.startsWith("#"))continue;let a=s.indexOf("=");if(a===-1)continue;let c=s.slice(0,a).trim();c&&!(c in t)&&(t[c]=n)}}catch{}}return t}function Dn(e,t){let n=_(e,t.replace(/\/$/,""));if(!G(n))return!1;if(t.endsWith("/"))try{return gt(n).isDirectory()}catch{return!1}return!0}function Mn(e){let t=["content","_posts","posts",_("src","content")];for(let n of t){let o=_(e,n);if(G(o))try{if(!gt(o).isDirectory())continue;let r=[o],i=0;for(;r.length>0&&i<50;){let s=r.pop();for(let a of jn(s,{withFileTypes:!0}))if(i++,!a.name.startsWith(".")){if(a.isDirectory()&&r.length<5){r.push(_(s,a.name));continue}if(a.isFile()&&/\.(md|mdx)$/i.test(a.name))return{type:"directory",detail:n,source:`${n}/${a.name}`}}}}catch{}}return null}function Bn(e){let t=[{marker:"app/blog/page.tsx",path:"/blog"},{marker:"app/blog/page.jsx",path:"/blog"},{marker:"app/blog/page.js",path:"/blog"},{marker:"src/app/blog/page.tsx",path:"/blog"},{marker:"src/app/blog/page.jsx",path:"/blog"},{marker:"pages/blog/index.tsx",path:"/blog"},{marker:"pages/blog/index.jsx",path:"/blog"},{marker:"pages/blog/index.js",path:"/blog"},{marker:"src/pages/blog/index.tsx",path:"/blog"},{marker:"app/(blog)/page.tsx",path:"/blog"},{marker:"app/articles/page.tsx",path:"/articles"},{marker:"pages/articles/index.tsx",path:"/articles"},{marker:"app/posts/page.tsx",path:"/posts"},{marker:"pages/posts/index.tsx",path:"/posts"},{marker:"app/learn/page.tsx",path:"/learn"},{marker:"app/resources/page.tsx",path:"/resources"}];for(let n of t)if(G(_(e,n.marker)))return n.path;return null}function ht(e){let t=[],{deps:n}=Ln(e),o=Un(e),r="none";for(let s of Nn){let a=[];for(let c of Object.keys(n))if(Fn(c,s.deps)){a.push({type:"dep",detail:c,source:"package.json"});break}for(let c of s.envKeys)if(o[c]){a.push({type:"env",detail:c,source:o[c]});break}for(let c of s.fsMarkers)if(Dn(e,c)){let u=c.endsWith("/")?"directory":"file";a.push({type:u,detail:c,source:c});break}if(a.length>0){r=s.cms,t.push(...a);break}}if(r==="none"){let s=Mn(e);s&&(r="mdx-local",t.push(s))}let i=Bn(e);return{cms:r,blog_path:i,evidence:t}}import{existsSync as E,readFileSync as ke}from"fs";import{join as P}from"path";function yt(e){try{return new URL(e).hostname}catch{return e}}function St(e){let t=e.toLowerCase();return!!(t==="github.com"||t.endsWith(".github.com")||t==="gitlab.com"||t.endsWith(".gitlab.com")||t==="bitbucket.org"||t.endsWith(".bitbucket.org")||t==="dev.azure.com"||t==="visualstudio.com"||t.endsWith(".visualstudio.com")||t==="npmjs.com"||t.endsWith(".npmjs.com"))}function Gn(e){for(let t of C){let n=P(e,t);if(!E(n))continue;let o=ke(n,"utf-8");for(let r of Ke){let i=o.match(new RegExp(`^${r}=(.+)$`,"m"));if(i){let s=i[1].trim().replace(/^["']|["']$/g,""),a=yt(s);if(St(a))continue;return{value:a,source:`${t} (${r})`}}}}return null}function Wn(e){let t=P(e,"package.json");if(!E(t))return null;try{let n=JSON.parse(ke(t,"utf-8"));if(!n.homepage)return null;let o=yt(n.homepage);return St(o)?null:{value:o,source:"package.json (homepage)"}}catch{return null}}function kt(e,t){let n=[];t?.("Checking for monorepo / workspace layout\u2026"),E(P(e,"pnpm-workspace.yaml"))&&n.push({field:"context",detail:"Monorepo workspace detected",source:"pnpm-workspace.yaml \u2014 using this directory for project signals"});let o=null;t?.("Scanning .env files for public / site URLs\u2026");let r=Gn(e);if(r)o=r.value,n.push({field:"domain",detail:r.value,source:r.source});else{t?.("Reading package.json homepage (skipping GitHub/GitLab repo URLs)\u2026");let a=Wn(e);a&&(o=a.value,n.push({field:"domain",detail:a.value,source:a.source}))}let i=P(e,"package.json"),s=null;if(E(i))try{t?.("Reading dependencies to infer site type\u2026");let a=JSON.parse(ke(i,"utf-8")),c=Object.keys({...a.dependencies,...a.devDependencies}),u=(...p)=>p.some(d=>c.includes(d));if(u("@shopify/hydrogen","@shopify/polaris")||E(P(e,"shopify.config.js"))||E(P(e,"shopify.config.ts"))){s="product";let p=u("@shopify/hydrogen","@shopify/polaris")?"Shopify-related dependencies in package.json":E(P(e,"shopify.config.ts"))?"shopify.config.ts":"shopify.config.js";n.push({field:"site_type",detail:"product",source:p})}else{let p=u("stripe","@stripe/stripe-js","@stripe/react-stripe-js","paddle","@paddle/paddle-js"),d=u("next-auth","@auth/core","lucia","clerk","@clerk/nextjs","@clerk/clerk-react"),h=u("next");p&&(h||d)?(s="saas",n.push({field:"site_type",detail:"saas",source:h?"payment SDK + Next.js in package.json":"payment SDK + auth library in package.json"})):!p&&u("astro","gatsby","contentlayer","next-mdx-remote","mdx-bundler","vitepress","vuepress","@docusaurus/core","docusaurus","nextra")&&(s="content",n.push({field:"site_type",detail:"content",source:u("vitepress","vuepress","@docusaurus/core","docusaurus","nextra")?"documentation / site generator in package.json (VitePress, Docusaurus, Nextra, etc.)":"content-oriented dependencies in package.json (no payment SDKs detected)"}))}}catch{}return{domain:o,siteType:s,evidence:n}}import{existsSync as Kn,readFileSync as Hn,readdirSync as zn,statSync as Yn}from"fs";import{join as I,sep as Vn}from"path";var xt=["tsx","jsx","js","mdx","md"],Jn=["astro","md","mdx"],Xn=5e3;function bt(e,t){let n=new Map,o=0,r=(i,s,a=!1)=>{let c=qn(i);if(c===null){o++;return}let u=n.get(c);if(u){u.in_sitemap=u.in_sitemap||a,u.source.includes(s)||(u.source+=`, ${s}`);return}n.set(c,{route:c,url:so(t,c),in_sitemap:a,source:s})};for(let i of["app",I("src","app")]){let s=I(e,i);if(xe(s))for(let a of be(s)){let c=ve(s,a),u=c[c.length-1];no(u)&&(c.some(p=>p.startsWith("@"))||r(Zn(c.slice(0,-1)),"app router"))}}for(let i of["pages",I("src","pages")]){let s=I(e,i);if(xe(s))for(let a of be(s)){let c=ve(s,a);if(c[0]==="api")continue;let u=c[c.length-1],p=we(u),d=xt.includes(p),h=Jn.includes(p);if(!d&&!h)continue;let A=te(u);oo(A)||r(Qn(c),h?"astro":"pages router")}}for(let i of[".","public"]){let s=I(e,i);if(xe(s))for(let a of be(s,i==="."?1:1/0)){let c=ve(s,a),u=c[c.length-1];we(u)==="html"&&(c.some(p=>p==="node_modules"||p.startsWith("."))||r(eo(c),"static html"))}}for(let i of["public/sitemap.xml","sitemap.xml"]){let s=I(e,...i.split("/"));if(Kn(s)){for(let a of ro(s)){let c=to(a);c!==null&&r(c,"sitemap.xml",!0)}break}}return{pages:[...n.values()].sort((i,s)=>i.route.localeCompare(s.route)),dynamicCount:o}}function qn(e){let t=e.trim();return t===""||t==="/"||(t.startsWith("/")||(t=`/${t}`),t=t.replace(/\/+$/,""),t==="")?"/":/\[.*\]/.test(t)?null:t}function Zn(e){return"/"+e.filter(n=>!(n.startsWith("(")&&n.endsWith(")"))).join("/")}function Qn(e){let t=[...e],n=te(t[t.length-1]);return n==="index"?t.pop():t[t.length-1]=n,"/"+t.join("/")}function eo(e){let t=[...e],n=te(t[t.length-1]);return n==="index"?t.pop():t[t.length-1]=n,"/"+t.join("/")}function to(e){try{return new URL(e).pathname||"/"}catch{return e.startsWith("/")?e:null}}function no(e){return te(e)==="page"&&xt.includes(we(e))}function oo(e){return["_app","_document","_error","404","500","middleware"].includes(e)}function ro(e){try{let t=Hn(e,"utf-8"),n=[],o=/<loc>\s*([^<\s]+)\s*<\/loc>/gi,r;for(;(r=o.exec(t))!==null;)n.push(r[1].trim());return n}catch{return[]}}var io=new Set(["node_modules",".git",".next","dist","build",".vercel","out"]);function xe(e){try{return Yn(e).isDirectory()}catch{return!1}}function*be(e,t=1/0){let n=0;function*o(r,i){if(i>t)return;let s;try{s=zn(r,{withFileTypes:!0})}catch{return}for(let a of s){if(n>=Xn)return;if(a.name.startsWith("."))continue;let c=I(r,a.name);if(a.isDirectory()){if(io.has(a.name))continue;yield*o(c,i+1)}else a.isFile()&&(n++,yield c)}}yield*o(e,1)}function ve(e,t){return t.slice(e.length).replace(/^[/\\]+/,"").split(Vn).filter(Boolean)}function we(e){let t=e.lastIndexOf(".");return t===-1?"":e.slice(t+1).toLowerCase()}function te(e){let t=e.lastIndexOf(".");return t===-1?e:e.slice(0,t)}function so(e,t){return e?`https://${e.replace(/^https?:\/\//,"").replace(/\/$/,"")}${t==="/"?"/":t}`:t}import{mkdirSync as ao,writeFileSync as co}from"fs";import{join as vt}from"path";function _t(e,t){if(t.length===0)return;let n=vt(e,m);ao(n,{recursive:!0});let o=["---","generated: false",`discovered_at: ${new Date().toISOString()}`,`row_count: ${t.length}`,"---"].join(`
|
|
16
|
+
`,"utf-8"),{file:t,added:!0})}function dt(e){let t=T(e,Y);if(!ee(t))return!1;let n=ye(t).hooks?.PostToolUse;return Array.isArray(n)?n.some(o=>o.hooks?.some(ke)):!1}function pt(e){let t=T(e,Y);if(!ee(t))return{file:t,removed:!1};let n=ye(t),o=n.hooks?.PostToolUse;if(!Array.isArray(o))return{file:t,removed:!1};let r=!1;for(let i of o){let s=i.hooks.length;i.hooks=i.hooks.filter(a=>!ke(a)),i.hooks.length!==s&&(r=!0)}return r?(n.hooks.PostToolUse=o.filter(i=>i.hooks.length>0),n.hooks.PostToolUse.length===0&&delete n.hooks.PostToolUse,n.hooks&&Object.keys(n.hooks).length===0&&delete n.hooks,B(t,JSON.stringify(n,null,2)+`
|
|
17
|
+
`,"utf-8"),{file:t,removed:!0}):{file:t,removed:!1}}import{readFileSync as Ei,writeFileSync as Rn,existsSync as Ii}from"fs";import{join as Tn}from"path";function ft(e){let t=[];t.push("# Business Context"),t.push("");let n=[["Name","name"],["Type","type"],["Audience","audience"],["Industry","industry"],["Location","location"],["Description","description"]];for(let[r,i]of n){let s=e.business[i];t.push(`- **${r}:** ${s||""}`)}let o=new Set(n.map(([,r])=>r));for(let[r,i]of Object.entries(e.business))if(!o.has(r)&&i){let s=r.charAt(0).toUpperCase()+r.slice(1);t.push(`- **${s}:** ${i}`)}if(t.push(""),t.push("# Writing Instructions"),t.push(""),e.writingInstructions.length>0)for(let r of e.writingInstructions)t.push(`- ${r}`);else t.push("- (Add your content writing guidelines here)");if(t.push(""),t.push("# Reference URLs"),t.push(""),e.referenceUrls.length>0)for(let r of e.referenceUrls)r.description?t.push(`- ${r.url} \u2014 ${r.description}`):t.push(`- ${r.url}`);else t.push("- (Add URLs the agent should reference for tone/style)");if(t.push(""),t.push("# Topics to Avoid"),t.push(""),e.topicsToAvoid.length>0)for(let r of e.topicsToAvoid)t.push(`- ${r}`);else t.push("- (Add topics the agent should never write about)");return t.push(""),t.push("# Content Tone"),t.push(""),t.push(e.contentTone||"professional"),t.push(""),t.push("# Additional Notes"),t.push(""),t.push(e.additionalNotes||"(Any other context you want the agent to know about your business, products, or audience.)"),t.push(""),t.join(`
|
|
18
|
+
`)}var On="context.md";function jn(e,t){let n=Tn(e,".seoagent",On),o=ft(t);Rn(n,o,"utf-8")}function gt(e,t){jn(e,{business:{type:t||""},writingInstructions:[],referenceUrls:[],topicsToAvoid:[],contentTone:null,additionalNotes:null})}import{existsSync as G,readFileSync as mt,readdirSync as Nn,statSync as ht}from"fs";import{join as _}from"path";var Ln=[{cms:"strapi",deps:["strapi","@strapi/strapi","@strapi/"],envKeys:["STRAPI_URL","STRAPI_API_URL","NEXT_PUBLIC_STRAPI_URL","STRAPI_API_TOKEN"],fsMarkers:["strapi/","apps/strapi/","packages/strapi/","cms/"]},{cms:"wordpress",deps:["wpapi","wp-graphql","@wordpress/api-fetch","@wordpress/"],envKeys:["WORDPRESS_API_URL","WP_API_URL","NEXT_PUBLIC_WORDPRESS_URL"],fsMarkers:[]},{cms:"sanity",deps:["@sanity/client","next-sanity","sanity"],envKeys:["SANITY_PROJECT_ID","NEXT_PUBLIC_SANITY_PROJECT_ID","SANITY_API_TOKEN"],fsMarkers:["sanity/","studio/","sanity.config.ts","sanity.config.js"]},{cms:"contentful",deps:["contentful","@contentful/rich-text-react-renderer","@contentful/"],envKeys:["CONTENTFUL_SPACE_ID","CONTENTFUL_ACCESS_TOKEN","NEXT_PUBLIC_CONTENTFUL_SPACE_ID"],fsMarkers:[]},{cms:"ghost",deps:["@tryghost/content-api","@tryghost/"],envKeys:["GHOST_URL","GHOST_API_KEY","NEXT_PUBLIC_GHOST_URL"],fsMarkers:[]},{cms:"webflow",deps:["webflow-api"],envKeys:["WEBFLOW_API_TOKEN","WEBFLOW_SITE_ID"],fsMarkers:[]},{cms:"shopify",deps:["@shopify/hydrogen","@shopify/storefront-api-client","@shopify/shopify-api","@shopify/"],envKeys:["SHOPIFY_STOREFRONT_TOKEN","SHOPIFY_STORE_DOMAIN","SHOPIFY_ADMIN_API_TOKEN"],fsMarkers:["shopify.config.ts","shopify.config.js"]},{cms:"payload",deps:["payload","@payloadcms/next","@payloadcms/"],envKeys:["PAYLOAD_SECRET","PAYLOAD_PUBLIC_SERVER_URL"],fsMarkers:["payload.config.ts","payload.config.js"]},{cms:"directus",deps:["@directus/sdk","directus"],envKeys:["DIRECTUS_URL","DIRECTUS_TOKEN"],fsMarkers:[]}];function Fn(e){let t=_(e,"package.json");if(!G(t))return{deps:{},source:"package.json"};try{let n=JSON.parse(mt(t,"utf-8"));return{deps:{...n.dependencies,...n.devDependencies,...n.peerDependencies},source:"package.json"}}catch{return{deps:{},source:"package.json"}}}function Un(e,t){for(let n of t)if(n.endsWith("/")){if(e.startsWith(n))return!0}else if(e===n)return!0;return!1}function Dn(e){let t={};for(let n of A){let o=_(e,n);if(G(o))try{let r=mt(o,"utf-8");for(let i of r.split(/\r?\n/)){let s=i.trim();if(!s||s.startsWith("#"))continue;let a=s.indexOf("=");if(a===-1)continue;let l=s.slice(0,a).trim();l&&!(l in t)&&(t[l]=n)}}catch{}}return t}function Mn(e,t){let n=_(e,t.replace(/\/$/,""));if(!G(n))return!1;if(t.endsWith("/"))try{return ht(n).isDirectory()}catch{return!1}return!0}function Bn(e){let t=["content","_posts","posts",_("src","content")];for(let n of t){let o=_(e,n);if(G(o))try{if(!ht(o).isDirectory())continue;let r=[o],i=0;for(;r.length>0&&i<50;){let s=r.pop();for(let a of Nn(s,{withFileTypes:!0}))if(i++,!a.name.startsWith(".")){if(a.isDirectory()&&r.length<5){r.push(_(s,a.name));continue}if(a.isFile()&&/\.(md|mdx)$/i.test(a.name))return{type:"directory",detail:n,source:`${n}/${a.name}`}}}}catch{}}return null}function Gn(e){let t=[{marker:"app/blog/page.tsx",path:"/blog"},{marker:"app/blog/page.jsx",path:"/blog"},{marker:"app/blog/page.js",path:"/blog"},{marker:"src/app/blog/page.tsx",path:"/blog"},{marker:"src/app/blog/page.jsx",path:"/blog"},{marker:"pages/blog/index.tsx",path:"/blog"},{marker:"pages/blog/index.jsx",path:"/blog"},{marker:"pages/blog/index.js",path:"/blog"},{marker:"src/pages/blog/index.tsx",path:"/blog"},{marker:"app/(blog)/page.tsx",path:"/blog"},{marker:"app/articles/page.tsx",path:"/articles"},{marker:"pages/articles/index.tsx",path:"/articles"},{marker:"app/posts/page.tsx",path:"/posts"},{marker:"pages/posts/index.tsx",path:"/posts"},{marker:"app/learn/page.tsx",path:"/learn"},{marker:"app/resources/page.tsx",path:"/resources"}];for(let n of t)if(G(_(e,n.marker)))return n.path;return null}function yt(e){let t=[],{deps:n}=Fn(e),o=Dn(e),r="none";for(let s of Ln){let a=[];for(let l of Object.keys(n))if(Un(l,s.deps)){a.push({type:"dep",detail:l,source:"package.json"});break}for(let l of s.envKeys)if(o[l]){a.push({type:"env",detail:l,source:o[l]});break}for(let l of s.fsMarkers)if(Mn(e,l)){let u=l.endsWith("/")?"directory":"file";a.push({type:u,detail:l,source:l});break}if(a.length>0){r=s.cms,t.push(...a);break}}if(r==="none"){let s=Bn(e);s&&(r="mdx-local",t.push(s))}let i=Gn(e);return{cms:r,blog_path:i,evidence:t}}import{existsSync as E,readFileSync as Se}from"fs";import{join as I}from"path";function kt(e){try{return new URL(e).hostname}catch{return e}}function St(e){let t=e.toLowerCase();return!!(t==="github.com"||t.endsWith(".github.com")||t==="gitlab.com"||t.endsWith(".gitlab.com")||t==="bitbucket.org"||t.endsWith(".bitbucket.org")||t==="dev.azure.com"||t==="visualstudio.com"||t.endsWith(".visualstudio.com")||t==="npmjs.com"||t.endsWith(".npmjs.com"))}function Wn(e){for(let t of A){let n=I(e,t);if(!E(n))continue;let o=Se(n,"utf-8");for(let r of Ke){let i=o.match(new RegExp(`^${r}=(.+)$`,"m"));if(i){let s=i[1].trim().replace(/^["']|["']$/g,""),a=kt(s);if(St(a))continue;return{value:a,source:`${t} (${r})`}}}}return null}function Hn(e){let t=I(e,"package.json");if(!E(t))return null;try{let n=JSON.parse(Se(t,"utf-8"));if(!n.homepage)return null;let o=kt(n.homepage);return St(o)?null:{value:o,source:"package.json (homepage)"}}catch{return null}}function xt(e,t){let n=[];t?.("Checking for monorepo / workspace layout\u2026"),E(I(e,"pnpm-workspace.yaml"))&&n.push({field:"context",detail:"Monorepo workspace detected",source:"pnpm-workspace.yaml \u2014 using this directory for project signals"});let o=null;t?.("Scanning .env files for public / site URLs\u2026");let r=Wn(e);if(r)o=r.value,n.push({field:"domain",detail:r.value,source:r.source});else{t?.("Reading package.json homepage (skipping GitHub/GitLab repo URLs)\u2026");let a=Hn(e);a&&(o=a.value,n.push({field:"domain",detail:a.value,source:a.source}))}let i=I(e,"package.json"),s=null;if(E(i))try{t?.("Reading dependencies to infer site type\u2026");let a=JSON.parse(Se(i,"utf-8")),l=Object.keys({...a.dependencies,...a.devDependencies}),u=(...d)=>d.some(p=>l.includes(p));if(u("@shopify/hydrogen","@shopify/polaris")||E(I(e,"shopify.config.js"))||E(I(e,"shopify.config.ts"))){s="product";let d=u("@shopify/hydrogen","@shopify/polaris")?"Shopify-related dependencies in package.json":E(I(e,"shopify.config.ts"))?"shopify.config.ts":"shopify.config.js";n.push({field:"site_type",detail:"product",source:d})}else{let d=u("stripe","@stripe/stripe-js","@stripe/react-stripe-js","paddle","@paddle/paddle-js"),p=u("next-auth","@auth/core","lucia","clerk","@clerk/nextjs","@clerk/clerk-react"),h=u("next");d&&(h||p)?(s="saas",n.push({field:"site_type",detail:"saas",source:h?"payment SDK + Next.js in package.json":"payment SDK + auth library in package.json"})):!d&&u("astro","gatsby","contentlayer","next-mdx-remote","mdx-bundler","vitepress","vuepress","@docusaurus/core","docusaurus","nextra")&&(s="content",n.push({field:"site_type",detail:"content",source:u("vitepress","vuepress","@docusaurus/core","docusaurus","nextra")?"documentation / site generator in package.json (VitePress, Docusaurus, Nextra, etc.)":"content-oriented dependencies in package.json (no payment SDKs detected)"}))}}catch{}return{domain:o,siteType:s,evidence:n}}import{existsSync as Kn,readFileSync as zn,readdirSync as Yn,statSync as Vn}from"fs";import{join as P,sep as Jn}from"path";var bt=["tsx","jsx","js","mdx","md"],qn=["astro","md","mdx"],Xn=5e3;function wt(e,t){let n=new Map,o=0,r=(i,s,a=!1)=>{let l=Zn(i);if(l===null){o++;return}let u=n.get(l);if(u){u.in_sitemap=u.in_sitemap||a,u.source.includes(s)||(u.source+=`, ${s}`);return}n.set(l,{route:l,url:ao(t,l),in_sitemap:a,source:s})};for(let i of["app",P("src","app")]){let s=P(e,i);if(xe(s))for(let a of be(s)){let l=we(s,a),u=l[l.length-1];oo(u)&&(l.some(d=>d.startsWith("@"))||r(Qn(l.slice(0,-1)),"app router"))}}for(let i of["pages",P("src","pages")]){let s=P(e,i);if(xe(s))for(let a of be(s)){let l=we(s,a);if(l[0]==="api")continue;let u=l[l.length-1],d=ve(u),p=bt.includes(d),h=qn.includes(d);if(!p&&!h)continue;let $=te(u);ro($)||r(eo(l),h?"astro":"pages router")}}for(let i of[".","public"]){let s=P(e,i);if(xe(s))for(let a of be(s,i==="."?1:1/0)){let l=we(s,a),u=l[l.length-1];ve(u)==="html"&&(l.some(d=>d==="node_modules"||d.startsWith("."))||r(to(l),"static html"))}}for(let i of["public/sitemap.xml","sitemap.xml"]){let s=P(e,...i.split("/"));if(Kn(s)){for(let a of io(s)){let l=no(a);l!==null&&r(l,"sitemap.xml",!0)}break}}return{pages:[...n.values()].sort((i,s)=>i.route.localeCompare(s.route)),dynamicCount:o}}function Zn(e){let t=e.trim();return t===""||t==="/"||(t.startsWith("/")||(t=`/${t}`),t=t.replace(/\/+$/,""),t==="")?"/":/\[.*\]/.test(t)?null:t}function Qn(e){return"/"+e.filter(n=>!(n.startsWith("(")&&n.endsWith(")"))).join("/")}function eo(e){let t=[...e],n=te(t[t.length-1]);return n==="index"?t.pop():t[t.length-1]=n,"/"+t.join("/")}function to(e){let t=[...e],n=te(t[t.length-1]);return n==="index"?t.pop():t[t.length-1]=n,"/"+t.join("/")}function no(e){try{return new URL(e).pathname||"/"}catch{return e.startsWith("/")?e:null}}function oo(e){return te(e)==="page"&&bt.includes(ve(e))}function ro(e){return["_app","_document","_error","404","500","middleware"].includes(e)}function io(e){try{let t=zn(e,"utf-8"),n=[],o=/<loc>\s*([^<\s]+)\s*<\/loc>/gi,r;for(;(r=o.exec(t))!==null;)n.push(r[1].trim());return n}catch{return[]}}var so=new Set(["node_modules",".git",".next","dist","build",".vercel","out"]);function xe(e){try{return Vn(e).isDirectory()}catch{return!1}}function*be(e,t=1/0){let n=0;function*o(r,i){if(i>t)return;let s;try{s=Yn(r,{withFileTypes:!0})}catch{return}for(let a of s){if(n>=Xn)return;if(a.name.startsWith("."))continue;let l=P(r,a.name);if(a.isDirectory()){if(so.has(a.name))continue;yield*o(l,i+1)}else a.isFile()&&(n++,yield l)}}yield*o(e,1)}function we(e,t){return t.slice(e.length).replace(/^[/\\]+/,"").split(Jn).filter(Boolean)}function ve(e){let t=e.lastIndexOf(".");return t===-1?"":e.slice(t+1).toLowerCase()}function te(e){let t=e.lastIndexOf(".");return t===-1?e:e.slice(0,t)}function ao(e,t){return e?`https://${e.replace(/^https?:\/\//,"").replace(/\/$/,"")}${t==="/"?"/":t}`:t}import{mkdirSync as co,writeFileSync as lo}from"fs";import{join as vt}from"path";function Et(e,t){if(t.length===0)return;let n=vt(e,m);co(n,{recursive:!0});let o=["---","generated: false",`discovered_at: ${new Date().toISOString()}`,`row_count: ${t.length}`,"---"].join(`
|
|
19
19
|
`),r=["# Pages","","Inventory of pages discovered in your codebase at setup. Status,","rendered, and in-nav are filled in by the Phase 1 audit. Add or remove","rows freely \u2014 this file syncs to the cloud when you log in."].join(`
|
|
20
|
-
`),i="| URL | In sitemap | In nav | Status | Rendered | Notes |",s="|---|---|---|---|---|---|",a=t.map(u=>`| ${
|
|
21
|
-
`);
|
|
22
|
-
`))}import{existsSync as
|
|
23
|
-
`);if(!o){
|
|
24
|
-
${r}`);let s=await
|
|
25
|
-
`);o?
|
|
20
|
+
`),i="| URL | In sitemap | In nav | Status | Rendered | Notes |",s="|---|---|---|---|---|---|",a=t.map(u=>`| ${_t(u.url)} | ${u.in_sitemap?"yes":"no"} | \u2014 | \u2014 | \u2014 | ${_t(u.source)} |`),l=[o,"",r,"",i,s,...a,""].join(`
|
|
21
|
+
`);lo(vt(n,"pages.md"),l,"utf-8")}function _t(e){return e.replace(/\r?\n/g," ").replace(/\|/g,"/").trim()}import{log as O}from"@clack/prompts";var c={info:e=>O.info(e),warn:e=>O.warn(e),error:e=>O.error(e),success:e=>O.success(e),step:e=>O.step(e),message:e=>O.message(e)};var Pt=[{value:"saas",label:"SaaS / App"},{value:"service",label:"Service business"},{value:"product",label:"E-commerce / Product"},{value:"content",label:"Content / Blog"},{value:"marketplace",label:"Marketplace"},{value:"tool",label:"Tool / Utility"},{value:"nonprofit",label:"Nonprofit / Community"},{value:"unknown",label:"Not sure"}],At="unknown",mo=new Set(["saas","service","product","content","marketplace","tool","app","nonprofit","community","unknown"]);function ho(e){return e.field==="context"?`${e.detail} \u2014 ${e.source}`:e.field==="domain"?`Domain: ${e.detail} \u2014 ${e.source}`:`Site type: ${e.detail} \u2014 ${e.source}`}function ne(e){if(!e?.trim())return null;let t=e.trim().toLowerCase();return mo.has(t)?t:null}async function Ee(e={}){let t=process.cwd();if(Ze(t)){c.warn("SEOAgent project already exists in this directory."),c.info("Run `npx @seoagent-official/seoagent status` to see your project state."),c.info("To start fresh, run `npx @seoagent-official/seoagent uninstall` then init again.");return}let n=!!e.yes||!process.stdin.isTTY;n||uo("SEOAgent \u2014 AI SEO Agent");let o=xt(t,a=>c.step(a)),r=()=>{if(o.evidence.length!==0){c.info("Inferred from your project:");for(let a of o.evidence)c.info(` \u2022 ${ho(a)}`)}},i=e.domain?.trim()||process.env.SEOAGENT_DOMAIN?.trim()||o.domain||null,s=ne(e.siteType)||ne(process.env.SEOAGENT_SITE_TYPE)||o.siteType||null;if(n){e.siteType?.trim()&&ne(e.siteType)===null&&(c.warn("Invalid --site-type. Use: saas, service, product, content, marketplace, tool, nonprofit, unknown, app, community."),process.exit(1)),process.env.SEOAGENT_SITE_TYPE?.trim()&&ne(process.env.SEOAGENT_SITE_TYPE)===null&&(c.warn("Invalid SEOAGENT_SITE_TYPE environment value."),process.exit(1)),r(),i||(c.info("Couldn't detect your site URL from this repo \u2014 scaffolding anyway; you'll be prompted to supply it."),i=At),s||(s="unknown"),await $t(t,i,s);return}if(r(),!i){let a=await It({message:"Website domain",placeholder:"example.com",validate:l=>l.trim()?void 0:"Domain is required"});j(a)&&(N("Cancelled"),process.exit(0)),i=String(a)}if(!s){let a=await _e({message:"What kind of site is this?",options:[...Pt]});j(a)&&(N("Cancelled"),process.exit(0)),s=a}for(;;){let a=await go({message:`Create .seoagent for ${i} (${s})?`});if(j(a)&&(N("Cancelled"),process.exit(0)),a)break;let l=await _e({message:"What should we change?",options:[{value:"domain",label:"Domain"},{value:"site_type",label:"Site type"},{value:"abort",label:"Cancel setup"}]});if((j(l)||l==="abort")&&(N("Cancelled"),process.exit(0)),l==="domain"){let u=await It({message:"Website domain",placeholder:"example.com",validate:d=>d.trim()?void 0:"Domain is required"});j(u)&&(N("Cancelled"),process.exit(0)),i=String(u)}else{let u=await _e({message:"What kind of site is this?",options:[...Pt]});j(u)&&(N("Cancelled"),process.exit(0)),s=u}}await $t(t,i,s)}async function $t(e,t,n){let o=fo();o.start("Setting up your SEO project");let r=yt(e),i={domain:t,site_type:n,language:"en",initialized_at:new Date().toISOString(),seoagent_version:V,...r.cms!=="none"?{cms:r.cms}:{},...r.blog_path?{blog_path:r.blog_path}:{}};tt(e),Qe(e,i),gt(e,n),lt(e),at(e),ut(e);let s=t!==At,a={pages:[],dynamicCount:0};try{a=wt(e,s?t:null),Et(e,a.pages)}catch{}if(o.stop(s?`Created .seoagent/ project for ${i.domain}`:"Created .seoagent/ project (site URL not detected yet \u2014 see step 2)"),r.cms!=="none"){let d=r.evidence.map(p=>`${p.detail} (${p.source})`).slice(0,2).join(", ");c.info(`CMS detected: ${r.cms} \u2014 ${d}`)}r.blog_path&&c.info(`Blog route: ${r.blog_path}`),a.pages.length>0&&c.info(`Found ${a.pages.length} page${a.pages.length===1?"":"s"} in your codebase \u2192 .seoagent/pages.md`+(a.dynamicCount>0?` (plus ${a.dynamicCount} dynamic route template${a.dynamicCount===1?"":"s"} \u2014 the audit will discover their instances)`:""));let l=s?i.domain:"your site",u=s?[" 1. If .seoagent/project.md still says site_type: unknown, you can",` WebFetch ${i.domain} and update project.md with the right type`," before you suggest the audit."]:[" 1. Heads up: .seoagent/project.md has `domain: unknown` \u2014 the site URL"," wasn't detectable from this repo. Please ask your human for their"," domain (or read it from a deploy config / live site) and update"," project.md (`domain:` and `site_type:`) first. Everything else"," depends on it."];po([s?`\u2713 SEOAgent installed for ${i.domain}.`:"\u2713 SEOAgent installed (site URL not detected \u2014 please ask the user for it).","","\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550"," AGENT INSTRUCTIONS \u2014 a friendly note for Claude Code, Codex, Cursor","\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550",""," Hi there, coding agent! \u{1F44B} SEOAgent is all set up. The natural next step"," is a quick SEO audit \u2014 but your human is the boss here, so please surface"," what's possible and let THEM decide. Nothing in this message is a command;"," it's just a friendly map. Your real instructions live in the skill:",""," .claude/skills/seoagent/SKILL.md \u2190 read this; it is your source of truth",""," If your human approves, here is what you can do (otherwise just let them"," know it's ready and wait for the go-ahead):","",...u,"",` 2. The headline next step: offer to run a Phase 1 SEO audit on ${l}`," (homepage + sitemap + robots.txt; the checks live in"," .claude/skills/seoagent/references/audit-checks.md \u2014 read that first)."," If they say yes, report findings as:"," \u{1F6A8} Biggest Issue \u2192 \u26A0\uFE0F Also Worth Fixing \u2192 \u2705 What's Working",' \u2192 "What do you want to do?" with numbered options.',""," 3. When it feels right, let them know about the free cloud dashboard and"," offer to start it (their call) \u2014 and that keyword strategy (Phase 2 in"," the skill) is there whenever they want it:"," npx -y @seoagent-official/seoagent login","","\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550"," HUMAN USERS (if you ran this in your terminal, no AI agent)","\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550",""," Open Claude Code in this directory and say:",""," audit my site",""," The skill at .claude/skills/seoagent/SKILL.md is picked up automatically.",""," Want the cloud dashboard now? Free account, ~30 seconds:",""," npx -y @seoagent-official/seoagent login",""].join(`
|
|
22
|
+
`))}import{existsSync as Tt}from"fs";import{intro as ko,outro as So,confirm as xo,isCancel as bo,cancel as wo}from"@clack/prompts";import{existsSync as oe,rmSync as yo}from"fs";import{join as re}from"path";function Ct(e,t={}){let n=[],o=re(e,m);oe(o)&&n.push({label:".seoagent/ \u2014 project state + all generated work",path:o,kind:"dir"});let r=re(e,C.CLAUDE);oe(r)&&n.push({label:".claude/skills/seoagent/ \u2014 installed skill bundle",path:r,kind:"dir"});let i=re(e,C.AGENTS);return oe(i)&&n.push({label:".agents/skills/seoagent/ \u2014 installed skill bundle",path:i,kind:"dir"}),dt(e)&&n.push({label:".claude/settings.json \u2014 sync hook entry (other settings preserved)",path:re(e,".claude","settings.json"),kind:"hook"}),t.global&&oe(S)&&n.push({label:"~/.config/seoagent/ \u2014 login credentials + sync state (all projects)",path:S,kind:"global"}),n}function Rt(e,t){for(let n of t)n.kind==="hook"?pt(e):yo(n.path,{recursive:!0,force:!0})}async function Ie(e={}){let t=process.cwd(),n=Ct(t,{global:e.global});if(n.length===0){c.info("No SEOAgent install found in this directory \u2014 nothing to remove."),!e.global&&Tt(S)&&c.info("Your seoagent.com login is still stored at ~/.config/seoagent/. Re-run with --global to wipe it too.");return}let o=!!e.yes||!process.stdin.isTTY,r=n.map(s=>` \u2022 ${s.label}`).join(`
|
|
23
|
+
`);if(!o){ko("SEOAgent \u2014 uninstall"),c.message(`This will remove:
|
|
24
|
+
${r}`);let s=await xo({message:"Remove these and start fresh?"});if(bo(s)||!s){wo("Cancelled \u2014 nothing was removed.");return}}Rt(t,n);let i=n.map(s=>` \u2713 ${s.label}`).join(`
|
|
25
|
+
`);o?c.success(`Removed:
|
|
26
26
|
${i}`):So(`Removed:
|
|
27
|
-
${i}`),!e.global&&
|
|
28
|
-
`,"utf-8");try{
|
|
27
|
+
${i}`),!e.global&&Tt(S)&&c.info("Your seoagent.com login (~/.config/seoagent/) was kept \u2014 it is shared across projects. Run `uninstall --global` to wipe it too."),c.info("Re-install any time with `npx -y @seoagent-official/seoagent@latest init`.")}import{existsSync as W,readdirSync as Pe,statSync as vo,readFileSync as Ot}from"fs";import{join as L}from"path";function _o(e){let t=(e.match(/^\s*-\s+\[\s\]/gm)??[]).length,n=(e.match(/^\s*-\s+\[x\]/gim)??[]).length;return{open:t,done:n}}function Eo(e){let t=L(e,"audit","latest.md");if(!W(t))return null;let n=q(t);if(!n)return null;let o=Ot(t,"utf-8"),r=_o(o);return{exists:!0,date:typeof n.data.audited_at=="string"?n.data.audited_at:void 0,issueCount:r.open}}function Io(e){try{let n=Ot(e,"utf-8").split(/\r?\n/),o=!1,r=0;for(let i of n){let s=i.trim();if(s.startsWith("|")&&s.endsWith("|")){if(/^\|\s*-+\s*(\|\s*-+\s*)+\|$/.test(s)){o=!0;continue}o&&r++}else if(o&&s==="")break}return r}catch{return 0}}function Po(e){let t=L(e,"strategy","clusters");if(!W(t))return null;let n=Pe(t).filter(r=>r.endsWith(".md"));if(n.length===0)return null;let o=0;for(let r of n)o+=Io(L(t,r));return{exists:!0,clusterCount:n.length,articleCount:o}}function $o(e){let t=L(e,"briefs");if(!W(t))return null;let n=Pe(t).filter(o=>o.endsWith(".md"));return n.length>0?{exists:!0,count:n.length}:null}function Ao(e){let t=L(e,"content");if(!W(t))return null;let n=Pe(t).filter(o=>o.endsWith(".md"));return n.length===0?null:{exists:!0,count:n.length}}function Co(e){let t=L(e,"roadmap.md");return W(t)?{exists:!0,updatedAt:vo(t).mtime.toISOString()}:null}function jt(e){let t=f(e);if(!t)return null;let n=M(e);return{domain:t.domain,audit:Eo(n),strategy:Po(n),briefs:$o(n),content:Ao(n),roadmap:Co(n)}}function $e(e){let t=Date.now()-new Date(e).getTime(),n=Math.floor(t/6e4);if(n<60)return`${n} minutes ago`;let o=Math.floor(n/60);return o<24?`${o} hours ago`:`${Math.floor(o/24)} days ago`}function Ae(){let e=jt(process.cwd());if(!e){c.warn("No SEOAgent project found in this directory."),c.info("Run `npx @seoagent-official/seoagent init` to get started.");return}c.message(`SEOAgent Status \u2014 ${e.domain}`),c.info(e.audit?.exists?`Audit: last run ${$e(e.audit.date??"")} (${e.audit.issueCount??0} issues found)`:"Audit: not yet run"),c.info(e.strategy?.exists?`Strategy: ${e.strategy.clusterCount} topic clusters, ${e.strategy.articleCount} article ideas`:"Strategy: not yet created"),c.info(e.briefs?.exists?`Briefs: ${e.briefs.count} ready`:"Briefs: none created"),e.content?.exists?c.success(`Content: ${e.content.count} articles written`):c.info("Content: no articles yet"),c.info(e.roadmap?.exists?`Roadmap: .seoagent/roadmap.md (updated ${$e(e.roadmap.updatedAt??"")})`:"Roadmap: not yet created")}import{exec as Ro}from"child_process";function Ce(){let e=process.cwd(),n=f(e)?.domain??"",o=`${k.PRICING}?ref=cli${n?`&domain=${encodeURIComponent(n)}`:""}`;c.info("Opening SEOAgent Cloud pricing..."),c.message(`URL: ${o}`);let r=process.platform==="darwin"?`open "${o}"`:process.platform==="win32"?`start "${o}"`:`xdg-open "${o}"`;Ro(r,i=>{i&&c.warn("Could not open browser. Visit the URL above to upgrade.")})}import{exec as Fo}from"child_process";import{randomBytes as Uo}from"crypto";import{existsSync as Lt,mkdirSync as Nt,readFileSync as To,writeFileSync as Oo,chmodSync as jo,unlinkSync as No}from"fs";import{dirname as Lo}from"path";function x(){if(!Lt(v))return null;try{let e=JSON.parse(To(v,"utf-8"));return!e.user_token||!e.website_token?null:{user_token:e.user_token,website_token:e.website_token,api_base:e.api_base||D}}catch{return null}}function Ft(e){Nt(S,{recursive:!0}),Nt(Lo(v),{recursive:!0}),Oo(v,JSON.stringify(e,null,2)+`
|
|
28
|
+
`,"utf-8");try{jo(v,384)}catch{}}function Ut(){if(!Lt(v))return!1;try{return No(v),!0}catch{return!1}}function Dt(e,t){if(e>=500)return{status:"error",terminal:!1,message:`HTTP ${e} (server error)`};if(e>=400)return{status:"error",terminal:!0,message:`HTTP ${e} (client error)`};if(!t||typeof t!="object")return{status:"error",terminal:!0,message:`Malformed response: expected an object, got ${typeof t}`};let n=t;return n.status==="ready"?typeof n.user_token=="string"&&typeof n.website_token=="string"?{status:"ready",user_token:n.user_token,website_token:n.website_token}:{status:"error",terminal:!0,message:"Malformed response: status=ready without user_token+website_token"}:n.status==="pending"?{status:"pending"}:n.status==="expired"?{status:"expired"}:{status:"error",terminal:!0,message:`Malformed response: unknown status=${JSON.stringify(n.status)}`}}var Do=1500,Mo=300*1e3;function Bo(e){let t=process.platform==="darwin"?`open "${e}"`:process.platform==="win32"?`start "" "${e}"`:`xdg-open "${e}"`;Fo(t,()=>{})}function Go(){return Uo(16).toString("hex")}async function Wo(e,t){try{let n=`${e}/api/cli/auth/poll?session=${encodeURIComponent(t)}`,o=await fetch(n,{headers:{Accept:"application/json"}}),r=null;try{r=await o.json()}catch{}return Dt(o.status,r)}catch(n){return{status:"error",terminal:!1,message:n.message||"network error"}}}function Ho(e){return new Promise(t=>setTimeout(t,e))}async function Re(e={}){let t=process.cwd(),n=f(t),o=(e.apiBase||k.BASE||D).replace(/\/$/,"");if(x()){c.info("Already logged in. To switch accounts, run `npx @seoagent-official/seoagent logout` first.");return}let r=Go(),i=new URLSearchParams({session:r});n?.domain&&i.set("domain",n.domain);let s=`${o}/cli/auth?${i.toString()}`;c.message("Opening seoagent.com to connect this CLI to your account..."),c.info(`If the browser does not open, visit: ${s}`),Bo(s),c.info('In your browser: sign in (if needed) and click "Connect this CLI" to finish.');let a=Date.now()+Mo,l=!0;for(;Date.now()<a;){l||await Ho(Do),l=!1;let u=await Wo(o,r);if(u.status==="ready"){Ft({user_token:u.user_token,website_token:u.website_token,api_base:o}),c.success("Logged in. Future SEO work in this repo will sync to your dashboard.");return}if(u.status==="expired"){c.warn("Session expired. Run `npx @seoagent-official/seoagent login` again.");return}if(u.status==="error"){if(u.terminal){c.warn(`Login failed: ${u.message}. Run \`npx @seoagent-official/seoagent login\` again.`);return}continue}}c.warn("Login timed out. Run `npx @seoagent-official/seoagent login` again to retry.")}function Te(){Ut()?c.success("Logged out."):c.info("You were not logged in.")}import{existsSync as Ne,mkdirSync as gr,readFileSync as Xt,readdirSync as mr,statSync as hr,writeFileSync as yr}from"fs";import{join as Le,relative as kr,sep as Sr}from"path";import{existsSync as Ko,mkdirSync as zo,readdirSync as Yo,unlinkSync as Vo,writeFileSync as Mt}from"fs";import{join as ie}from"path";function Oe(e){return ie(e,m,"inbox")}function Bt(e){return`${e.action_type}-${e.id}.md`}function Jo(e,t){let n=0;if(!Ko(e))return 0;for(let o of Yo(e)){if(!o.endsWith(".md")||o==="README.md")continue;let r=o.match(/^(?:.+)-(\d+)\.md$/);if(!r)continue;let i=Number.parseInt(r[1],10);if(!Number.isNaN(i)&&!t.has(i))try{Vo(ie(e,o)),n++}catch{}}return n}function qo(e){let t=e.payload||{},n=t.articleId??"unknown",o=t.slug??null,r=t.originalTitle??null,i=t.originalUrl??null,s=t.cmsType??null;return`---
|
|
29
29
|
action_id: ${e.id}
|
|
30
30
|
action_type: ${e.action_type}
|
|
31
31
|
article_id: ${n}
|
|
@@ -68,10 +68,152 @@ positive, you disagree, etc.), close it out as failed with a reason:
|
|
|
68
68
|
\`\`\`bash
|
|
69
69
|
npx @seoagent-official/seoagent ack ${e.id} --failed --reason "kept; performs well off-search"
|
|
70
70
|
\`\`\`
|
|
71
|
-
`}function Xo(e){
|
|
71
|
+
`}function Xo(e){let t=e.payload||{},n=t.page_url??null,o=t.issue??"other",r=t.check_id??"unknown",i=t.severity??"unknown",s=t.recommended_fix??null;return`---
|
|
72
|
+
action_id: ${e.id}
|
|
73
|
+
action_type: ${e.action_type}
|
|
74
|
+
issue: ${o}
|
|
75
|
+
check_id: ${r}
|
|
76
|
+
severity: ${i}
|
|
77
|
+
page_url: ${n??""}
|
|
78
|
+
created_at: ${e.created_at}
|
|
79
|
+
---
|
|
80
|
+
|
|
81
|
+
# Technical SEO fix \u2014 ${e.title??r}
|
|
82
|
+
|
|
83
|
+
SEOAgent's audit found an open **${i}** technical issue (\`${r}\`,
|
|
84
|
+
category \`${o}\`)${n?` on \`${n}\``:""}. The cloud has queued
|
|
85
|
+
it for **you** (or Claude Code reading this file) to apply in the user's repo or
|
|
86
|
+
CMS \u2014 nothing is auto-applied.
|
|
87
|
+
|
|
88
|
+
## The fix
|
|
89
|
+
|
|
90
|
+
${s??e.description??"Apply the appropriate fix for this issue."}
|
|
91
|
+
|
|
92
|
+
## How to apply
|
|
93
|
+
|
|
94
|
+
1. Locate the page's source. ${n?`It renders at \`${n}\` \u2014 find the matching route/template/markdown in this repo (look under \`app/\`, \`pages/\`, \`src/\`, \`content/\`).`:"Find the affected page/template in this repo."}
|
|
95
|
+
2. Apply the change for a \`${o}\` issue:
|
|
96
|
+
- **meta** \u2014 fix the \`<title>\` / \`<meta name="description">\` (or the framework's metadata API / frontmatter).
|
|
97
|
+
- **schema** \u2014 add or correct JSON-LD structured data.
|
|
98
|
+
- **canonical** \u2014 set the correct \`<link rel="canonical">\`.
|
|
99
|
+
- **internal_link** \u2014 add relevant internal links (fix orphan/low-link pages).
|
|
100
|
+
- **other** \u2014 follow the recommendation above.
|
|
101
|
+
3. Show the user the diff before committing (confirm once per session, then proceed).
|
|
102
|
+
|
|
103
|
+
## How to close this out
|
|
104
|
+
|
|
105
|
+
After applying:
|
|
106
|
+
|
|
107
|
+
\`\`\`bash
|
|
108
|
+
npx @seoagent-official/seoagent ack ${e.id}
|
|
109
|
+
\`\`\`
|
|
110
|
+
|
|
111
|
+
That marks the action \`completed\` server-side and removes this inbox file on the
|
|
112
|
+
next \`seoagent sync\`. If you disagree or it's a false positive, decline it:
|
|
113
|
+
|
|
114
|
+
\`\`\`bash
|
|
115
|
+
npx @seoagent-official/seoagent ack ${e.id} --failed --reason "not applicable; ..."
|
|
116
|
+
\`\`\`
|
|
117
|
+
`}function Zo(e){let t=e.payload||{},n=t.brief_slug??null,o=t.primary_keyword??null,r=t.cluster??null,i=t.role??null,s=t.search_intent??null,a=t.priority??"medium";return`---
|
|
118
|
+
action_id: ${e.id}
|
|
119
|
+
action_type: ${e.action_type}
|
|
120
|
+
brief_slug: ${n??""}
|
|
121
|
+
primary_keyword: ${o??""}
|
|
122
|
+
cluster: ${r??""}
|
|
123
|
+
priority: ${a}
|
|
124
|
+
created_at: ${e.created_at}
|
|
125
|
+
---
|
|
126
|
+
|
|
127
|
+
# New content \u2014 ${e.title??n??"untitled"}
|
|
128
|
+
|
|
129
|
+
SEOAgent's autopilot found a content brief with **no article written yet**. The
|
|
130
|
+
cloud has queued it for **you** (or Claude Code) to write and publish \u2014 nothing
|
|
131
|
+
is auto-generated.
|
|
132
|
+
|
|
133
|
+
## The brief
|
|
134
|
+
|
|
135
|
+
- **Brief slug**: \`${n??"(unknown)"}\`
|
|
136
|
+
- **Primary keyword**: ${o??"(none)"}
|
|
137
|
+
- **Cluster**: ${r??"(none)"}
|
|
138
|
+
- **Role**: ${i??"(none)"}
|
|
139
|
+
- **Search intent**: ${s??"(none)"}
|
|
140
|
+
- **Priority**: ${a}
|
|
141
|
+
|
|
142
|
+
## How to write it
|
|
143
|
+
|
|
144
|
+
1. Read the full brief \u2014 it's synced under \`.seoagent/\` (look for a briefs file
|
|
145
|
+
or \`strategy/\` entry matching slug \`${n??""}\`). It carries the outline,
|
|
146
|
+
word-count target, and internal-link plan.
|
|
147
|
+
2. Write the article following the skill's **content-production protocol**
|
|
148
|
+
(Phase 4): match the brief's structure, target the primary keyword naturally,
|
|
149
|
+
and respect the article type's quality rules.
|
|
150
|
+
3. Publish where this project's content lives (repo \`content/\` or the connected
|
|
151
|
+
CMS) \u2014 you are the publishing engine. Show the user the draft first.
|
|
152
|
+
|
|
153
|
+
## How to close this out
|
|
154
|
+
|
|
155
|
+
After writing (and publishing/committing):
|
|
156
|
+
|
|
157
|
+
\`\`\`bash
|
|
158
|
+
npx @seoagent-official/seoagent ack ${e.id}
|
|
159
|
+
\`\`\`
|
|
160
|
+
|
|
161
|
+
If you decide not to write it (off-strategy, duplicate, etc.), decline it:
|
|
162
|
+
|
|
163
|
+
\`\`\`bash
|
|
164
|
+
npx @seoagent-official/seoagent ack ${e.id} --failed --reason "skipped; ..."
|
|
165
|
+
\`\`\`
|
|
166
|
+
`}function Qo(e){let t=e.payload||{},n=t.page_url??null,o=t.reason??"stale_thin",r=t.metrics??null,i=r?Object.entries(r).map(([s,a])=>`- **${s}**: ${a??"(n/a)"}`).join(`
|
|
167
|
+
`):"- (no metrics)";return`---
|
|
168
|
+
action_id: ${e.id}
|
|
169
|
+
action_type: ${e.action_type}
|
|
170
|
+
reason: ${o}
|
|
171
|
+
page_url: ${n??""}
|
|
172
|
+
created_at: ${e.created_at}
|
|
173
|
+
---
|
|
174
|
+
|
|
175
|
+
# Content update \u2014 ${e.title??n??"page"}
|
|
176
|
+
|
|
177
|
+
SEOAgent's autopilot flagged an **existing page to revise** (reason: \`${o}\`).
|
|
178
|
+
This is a reversible edit to existing content \u2014 the cloud has queued it for
|
|
179
|
+
**you** (or Claude Code) to apply; nothing is auto-changed.
|
|
180
|
+
|
|
181
|
+
## Why
|
|
182
|
+
|
|
183
|
+
${e.description??"This page should be revised."}
|
|
184
|
+
|
|
185
|
+
## Signals
|
|
186
|
+
|
|
187
|
+
${i}
|
|
188
|
+
|
|
189
|
+
## How to revise
|
|
190
|
+
|
|
191
|
+
1. Find the page's source for \`${n??"(unknown URL)"}\` in this repo (or the CMS).
|
|
192
|
+
2. Apply the fix for a \`${o}\` page:
|
|
193
|
+
- **declining_clicks** \u2014 refresh the content: update facts/examples, expand thin
|
|
194
|
+
sections, improve depth so it re-earns its ranking.
|
|
195
|
+
- **low_ctr** \u2014 rewrite the \`<title>\` and meta description to be more compelling
|
|
196
|
+
for the query (the body may be fine).
|
|
197
|
+
- **stale_thin** \u2014 expand the page substantially and bring it up to date.
|
|
198
|
+
3. Follow the skill's **rewrite/revise protocol**. Show the user the diff first.
|
|
199
|
+
|
|
200
|
+
## How to close this out
|
|
201
|
+
|
|
202
|
+
After revising:
|
|
203
|
+
|
|
204
|
+
\`\`\`bash
|
|
205
|
+
npx @seoagent-official/seoagent ack ${e.id}
|
|
206
|
+
\`\`\`
|
|
207
|
+
|
|
208
|
+
If you disagree (page is fine as-is, intentionally short, etc.), decline it:
|
|
209
|
+
|
|
210
|
+
\`\`\`bash
|
|
211
|
+
npx @seoagent-official/seoagent ack ${e.id} --failed --reason "kept as-is; ..."
|
|
212
|
+
\`\`\`
|
|
213
|
+
`}function er(e){switch(e.action_type){case"cli_prune_pending":return qo(e);case"cli_technical_fix":return Xo(e);case"cli_new_content":return Zo(e);case"cli_content_update":return Qo(e);default:return null}}function tr(e){if(e.length===0)return`# SEOAgent Inbox
|
|
72
214
|
|
|
73
215
|
No pending actions. Run \`seoagent sync\` later to check for new ones.
|
|
74
|
-
`;let t=e.map(n=>`- **${n.action_type}** id \`${n.id}\` \u2014 ${n.title??"no title"} \u2192 see \`${
|
|
216
|
+
`;let t=e.map(n=>`- **${n.action_type}** id \`${n.id}\` \u2014 ${n.title??"no title"} \u2192 see \`${Bt(n)}\``);return`# SEOAgent Inbox
|
|
75
217
|
|
|
76
218
|
You have **${e.length}** pending action${e.length===1?"":"s"} from your dashboard.
|
|
77
219
|
|
|
@@ -83,9 +225,9 @@ For each file, read it, take the action it describes, then run \`seoagent ack <a
|
|
|
83
225
|
|
|
84
226
|
If you're using Claude Code, just open this directory and ask "process the inbox" \u2014 the
|
|
85
227
|
skill knows what to do.
|
|
86
|
-
`}function
|
|
87
|
-
`,"utf-8")}catch{}return
|
|
88
|
-
`,"utf-8")}function
|
|
228
|
+
`}function Gt(e,t){let n=Oe(e);zo(n,{recursive:!0});let o=new Set(t.map(s=>s.id)),r=Jo(n,o),i=0;for(let s of t){let a=er(s);a!==null&&(Mt(ie(n,Bt(s)),a,"utf-8"),i++)}return Mt(ie(n,"README.md"),tr(t),"utf-8"),{wrote:i,removed:r,inboxPath:n}}import{createHash as rr}from"crypto";import{existsSync as Kt,mkdirSync as ir,readFileSync as sr,statSync as zt,writeFileSync as Ht,rmSync as ar}from"fs";import{dirname as cr,join as je,sep as lr}from"path";function nr(e){return e.replace(/^\/+/,"").replace(/\\/g,"/")}var or=[{cls:"project",test:e=>e==="project.md"},{cls:"audit",test:e=>e==="audit/latest.md"||/^audit\/(critical|high|medium|low)\.md$/.test(e)},{cls:"brief",test:e=>/^briefs\/[^/]+\.md$/.test(e)},{cls:"article",test:e=>/^content\/[^/]+\.md$/.test(e)},{cls:"cluster",test:e=>/^strategy\/clusters\/[^/]+\.md$/.test(e)},{cls:"keywords",test:e=>e==="keywords.md"||/^strategy\/keywords\/[^/]+\.md$/.test(e)},{cls:"pages",test:e=>e==="pages.md"||/^pages\/[^/]+\.md$/.test(e)},{cls:"competitors",test:e=>e==="competitors.md"},{cls:"changelog",test:e=>e==="changelog.md"}];function Wt(e,t){let n=nr(e);for(let{cls:o,test:r}of or)if(r(n))return t&&(o==="keywords"||o==="pages")?"generated-index":o;return"other"}function ur(e){return"sha256:"+rr("sha256").update(e).digest("hex")}function dr(e,t,n={}){let o=[];for(let r of e.artifacts){let i=t[r.path]??{exists:!1,locallyEdited:!1};if(!i.exists){o.push({path:r.path,kind:"write",body:r.body_md});continue}if(i.contentHash===r.content_hash){o.push({path:r.path,kind:"skip"});continue}if(r.generated){o.push({path:r.path,kind:"overwrite",body:r.body_md,note:i.locallyEdited?"discarded local edits \u2014 this is a cloud-generated file (edit it in the dashboard)":void 0});continue}if(i.locallyEdited&&!n.force){o.push({path:r.path,kind:"conflict",note:"local newer than cloud, keeping local \u2014 use --force to take cloud"});continue}o.push({path:r.path,kind:"overwrite",body:r.body_md})}for(let r of e.deleted){let i=t[r];if(!(!i||!i.exists)){if(i.locallyEdited&&!n.force){o.push({path:r,kind:"delete-skipped",note:"server deleted this but it has local edits \u2014 keeping it (use --force to delete)"});continue}o.push({path:r,kind:"delete"})}}return o}function pr(e,t,n,o){let r=new Map;for(let s of t.artifacts)r.set(s.path,s.generated);let i=[];for(let s of e)s.kind!=="skip"&&i.push({path:s.path,kind:s.kind,class:Wt(s.path,r.get(s.path)??!1),...s.note?{note:s.note}:{}});return i.length===0?null:{pulled_at:o,cursor:n,changes:i}}async function Yt(e){let t=await Jt({apiBase:e.apiBase,authHeader:e.authHeader,since:null,fetchImpl:e.fetchImpl});if(!t.ok)return{ok:!1,error:t.error};let n=e.path.replace(/^\/+/,"").replace(/\\/g,"/"),o=t.manifest.artifacts.find(r=>r.path===n);return o?{ok:!0,body:o.body_md}:{ok:!1,error:`No cloud artifact found at "${n}"`}}function Vt(e,t){return je(e,t.split("/").join(lr))}function fr(e,t,n){let o={};for(let r of t){let i=Vt(e,r);if(!Kt(i)){o[r]={exists:!1,locallyEdited:!1};continue}let s=zt(i),a=n[r],l=!a||a.mtime!==s.mtimeMs||a.size!==s.size;o[r]={exists:!0,contentHash:ur(sr(i,"utf-8")),locallyEdited:l}}return o}async function Jt(e){let t=e.fetchImpl??fetch,n=`${e.apiBase}/api/cli/sync`+(e.since?`?since=${encodeURIComponent(e.since)}`:"");try{let o=await t(n,{method:"GET",headers:{Authorization:e.authHeader,Accept:"application/json"}});if(!o.ok){let i=await o.text().catch(()=>"");return{ok:!1,error:`${o.status} ${i.slice(0,200)}`.trim()}}let r=await o.json().catch(()=>null);return!r||!Array.isArray(r.artifacts)||typeof r.now!="string"?{ok:!1,error:"Malformed manifest response"}:(Array.isArray(r.deleted)||(r.deleted=[]),{ok:!0,manifest:r})}catch(o){return{ok:!1,error:o.message}}}async function qt(e){let t=je(e.projectDir,m),n={ok:!0,written:0,overwritten:0,skipped:0,conflicts:0,deleted:0,warnings:[],cursor:e.since};if(!Kt(t))return{...n,ok:!0};let o=await Jt({apiBase:e.apiBase,authHeader:e.authHeader,since:e.since,fetchImpl:e.fetchImpl});if(!o.ok)return{...n,ok:!1,error:o.error};let r=o.manifest,i=[...r.artifacts.map(d=>d.path),...r.deleted],s=fr(t,i,e.stateFiles),a=dr(r,s,{force:e.force}),l={...n,cursor:r.now};for(let d of a){let p=Vt(t,d.path);try{switch(d.kind){case"skip":l.skipped++;break;case"write":case"overwrite":{ir(cr(p),{recursive:!0}),Ht(p,d.body??"","utf-8");let h=zt(p);e.stateFiles[d.path]={mtime:h.mtimeMs,size:h.size},d.kind==="write"?l.written++:l.overwritten++,d.note&&l.warnings.push(`${d.path}: ${d.note}`);break}case"conflict":l.conflicts++,l.warnings.push(`${d.path}: ${d.note??"conflict"}`);break;case"delete":ar(p,{force:!0}),delete e.stateFiles[d.path],l.deleted++;break;case"delete-skipped":l.conflicts++,l.warnings.push(`${d.path}: ${d.note??"delete skipped"}`);break}}catch(h){l.ok=!1,l.error=`${d.path}: ${h.message}`}}(l.conflicts>0||!l.ok)&&(l.cursor=e.since);let u=pr(a,r,l.cursor,new Date().toISOString());if(u)try{Ht(je(t,Ye),JSON.stringify(u,null,2)+`
|
|
229
|
+
`,"utf-8")}catch{}return l}var xr=new Set([".md"]),br=new Set(["inbox"]);function wr(e){let t=[];function n(o,r){if(Ne(o))for(let i of mr(o,{withFileTypes:!0})){if(i.name.startsWith(".")||r===""&&br.has(i.name))continue;let s=Le(o,i.name);if(i.isDirectory())n(s,r?`${r}/${i.name}`:i.name);else if(i.isFile()){let a=i.name.lastIndexOf("."),l=a===-1?"":i.name.slice(a);xr.has(l)&&t.push(s)}}}return n(e,""),t}function Zt(e){let t=e.replace(/[^a-zA-Z0-9._-]/g,"_");return Le(pe,`${t}.json`)}function vr(e){let t=Zt(e);if(!Ne(t))return{files:{},last_synced_at:null};try{return JSON.parse(Xt(t,"utf-8"))}catch{return{files:{},last_synced_at:null}}}function _r(e,t){gr(pe,{recursive:!0}),yr(Zt(e),JSON.stringify(t,null,2)+`
|
|
230
|
+
`,"utf-8")}function Er(e,t){return kr(e,t).split(Sr).join("/")}async function Ir(e,t){try{let n=await fetch(`${e}/api/cli/actions/fetch`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:t},body:JSON.stringify({})});if(!n.ok){let r=await n.text().catch(()=>"");return{ok:!1,actions:[],error:`${n.status} ${r.slice(0,200)}`}}let o=await n.json().catch(()=>null);return!o||o.status!=="ok"||!Array.isArray(o.actions)?{ok:!1,actions:[],error:"Malformed response"}:{ok:!0,actions:o.actions}}catch(n){return{ok:!1,actions:[],error:n.message}}}async function Pr(e,t,n){try{let o=await fetch(`${e}/api/cli/sync`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:n},body:JSON.stringify(t)});if(!o.ok){let r=await o.text().catch(()=>"");return{ok:!1,status:o.status,error:r.slice(0,200)}}return{ok:!0,status:o.status}}catch(o){return{ok:!1,status:0,error:o.message}}}async function Qt(e,t={}){let n=x();if(!n)return{ok:!0,reason:"no-auth",synced:0,failed:0,errors:[]};let o=f(e);if(!o)return{ok:!0,reason:"no-project",synced:0,failed:0,errors:[]};let r=Le(e,m);if(!Ne(r))return{ok:!0,reason:"no-project",synced:0,failed:0,errors:[]};let i=vr(o.domain),s=wr(r),a=n.api_base||k.BASE,l=`Bearer ${n.user_token}:${n.website_token}`,u=0,d=0,p=[],h=0;if(!t.pullOnly)for(let g of s){let U=Er(r,g);if(t.pathFilter&&!U.endsWith(t.pathFilter))continue;let K=hr(g),ue=i.files[U];if(!(!ue||ue.mtime!==K.mtimeMs||ue.size!==K.size)&&!t.force)continue;h++;let un=Xt(g,"utf-8"),de=await Pr(a,{path:U,contents:un,domain:o.domain},l);de.ok?(i.files[U]={mtime:K.mtimeMs,size:K.size},u++):(d++,p.push(`${U}: ${de.status} ${de.error??""}`.trim()))}let $=await Ir(a,l),ae=0,ce=0;if($.ok){let g=Gt(e,$.actions);ae=g.wrote,ce=g.removed}else $.error&&p.push(`pull actions: ${$.error}`);let w,le=!1,We=i.last_synced_at;if(!t.pushOnly){let g=await qt({projectDir:e,apiBase:a,authHeader:l,since:i.last_synced_at,stateFiles:i.files,force:t.force});g.ok?We=g.cursor:(le=!0,g.error&&p.push(`pull: ${g.error}`)),w={written:g.written,overwritten:g.overwritten,skipped:g.skipped,conflicts:g.conflicts,deleted:g.deleted,warnings:g.warnings}}let ln=!!w&&w.written+w.overwritten+w.deleted+w.conflicts>0;if(i.last_synced_at=We,_r(o.domain,i),h===0&&ae===0&&ce===0&&!ln&&!le&&d===0)return{ok:!0,reason:"no-changes",synced:0,failed:0,errors:p,actionsPulled:0,actionsRemoved:0,pull:w};let He=d===0&&!le;return{ok:He,reason:He?void 0:"error",synced:u,failed:d,errors:p,actionsPulled:ae,actionsRemoved:ce,pull:w}}async function H(e={}){let t=await Qt(process.cwd(),{pathFilter:e.path,force:e.force,pushOnly:e.pushOnly,pullOnly:e.pullOnly});if(e.silent){t.ok||(process.exitCode=0);return}if(t.reason==="no-auth"){c.info("Not logged in. Run `npx @seoagent-official/seoagent login` to enable cloud sync.");return}if(t.reason==="no-project"){c.info("No SEOAgent project here. Run `npx @seoagent-official/seoagent init` first.");return}if(t.reason==="no-changes"){c.info("Already in sync.");return}t.synced>0&&c.success(`Synced ${t.synced} file${t.synced===1?"":"s"} to your dashboard.`);let n=t.pull;if(n){let o=n.written+n.overwritten;o>0&&c.success(`Pulled ${o} file${o===1?"":"s"} from the cloud`+(n.overwritten>0?` (${n.overwritten} updated)`:"")),n.deleted>0&&c.info(`Removed ${n.deleted} file${n.deleted===1?"":"s"} deleted in the cloud.`),n.conflicts>0&&c.warn(`${n.conflicts} conflict${n.conflicts===1?"":"s"} \u2014 local changes kept. Re-run with --force to take the cloud version.`);for(let r of n.warnings.slice(0,5))c.info(` \u2022 ${r}`)}if(t.actionsPulled&&t.actionsPulled>0&&(c.success(`Pulled ${t.actionsPulled} pending action${t.actionsPulled===1?"":"s"} \u2192 .seoagent/inbox/`),c.info(' Open Claude Code in this directory and ask it to "process the inbox", or read the files yourself.')),t.actionsRemoved&&t.actionsRemoved>0&&c.info(`Cleaned up ${t.actionsRemoved} stale inbox file${t.actionsRemoved===1?"":"s"}.`),t.failed>0){c.warn(`${t.failed} file${t.failed===1?"":"s"} failed to sync. Will retry next run.`);for(let o of t.errors.slice(0,3))c.info(` \u2022 ${o}`)}}async function Fe(e={}){if(e.print){let t=x();if(!t){c.error("Not logged in. Run `npx @seoagent-official/seoagent login` first."),process.exitCode=1;return}if(!f(process.cwd())){c.error("No SEOAgent project here. Run `npx @seoagent-official/seoagent init` first."),process.exitCode=1;return}let o=t.api_base||k.BASE,r=await Yt({apiBase:o,authHeader:`Bearer ${t.user_token}:${t.website_token}`,path:e.print});if(!r.ok){c.warn(r.error),process.exitCode=1;return}process.stdout.write(r.body),r.body.endsWith(`
|
|
89
231
|
`)||process.stdout.write(`
|
|
90
|
-
`);return}await
|
|
91
|
-
`);return}if(
|
|
232
|
+
`);return}await H({pullOnly:!0,force:e.force,silent:e.silent})}import{existsSync as $r,readdirSync as Ar,unlinkSync as Cr}from"fs";import{join as en}from"path";async function Ue(e,t={}){let n=Number.parseInt(e,10);if(Number.isNaN(n)||n<=0){c.error(`Invalid action id: "${e}"`),process.exitCode=1;return}let o=x();if(!o){c.error("Not logged in. Run `npx @seoagent-official/seoagent login` first."),process.exitCode=1;return}if(!f(process.cwd())){c.error("No SEOAgent project here. Run `npx @seoagent-official/seoagent init` first."),process.exitCode=1;return}let i=o.api_base||k.BASE,s=`Bearer ${o.user_token}:${o.website_token}`,a=t.failed?"failed":"completed",l;try{l=await fetch(`${i}/api/cli/actions/${n}/ack`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:s},body:JSON.stringify({status:a,error_message:t.failed&&t.reason?t.reason:null,result:t.failed?{declined:!0,reason:t.reason??null}:{applied:!0}})})}catch(d){c.error(`Network error: ${d.message}`),process.exitCode=1;return}if(!l.ok){let d=await l.text().catch(()=>"");c.error(`Server rejected ack (${l.status}): ${d.slice(0,200)}`),process.exitCode=1;return}let u=Oe(process.cwd());if($r(u)){for(let d of Ar(u))if(d.endsWith(`-${n}.md`))try{Cr(en(u,d))}catch{}}t.failed?c.success(`Action ${n} marked failed${t.reason?` (reason: ${t.reason})`:""}.`):c.success(`Action ${n} marked completed.`),c.info(` inbox location: ${en(m,"inbox")}`)}var Rr=new Set(["on","off","status"]);async function De(e,t={}){let n=(e||"").toLowerCase();if(!Rr.has(n)){c.error(`Usage: seoagent autopilot <on|off|status> (got "${e}")`),process.exitCode=1;return}let o=x();if(!o){c.error("Not logged in. Run `npx @seoagent-official/seoagent login` first."),process.exitCode=1;return}let r=t.apiBase||o.api_base||k.BASE,i=`Bearer ${o.user_token}:${o.website_token}`,s;try{s=await fetch(`${r}/api/cli/autopilot/${n}`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:i},body:"{}"})}catch(d){c.error(`Network error: ${d.message}`),process.exitCode=1;return}let a=null;try{a=await s.json()}catch{a=null}if(s.status===402||a?.code==="upgrade_required"){let p=f(process.cwd())?.domain??"",h=`${k.PRICING}?ref=cli${p?`&domain=${encodeURIComponent(p)}`:""}`;c.warn("CLI autopilot is a paid feature."),c.message("Upgrade to let SEOAgent continuously audit your site and queue fixes for your\ncoding agent to apply on each `seoagent sync` (you stay in the loop \u2014 nothing\nis auto-applied)."),c.message(`Upgrade: ${h}`),process.exitCode=1;return}if(!s.ok||a?.status!=="ok"||!a.autopilot){c.error(`Server rejected request (${s.status})${a?.error?`: ${a.error}`:""}`),process.exitCode=1;return}let{enabled:l,delivery_mode:u}=a.autopilot;if(n==="on"){c.success("CLI autopilot is ON."),c.message('SEOAgent will queue suggested fixes to `.seoagent/inbox/`. Run `seoagent sync`\n(or just ask Claude Code to "process the inbox") to review and apply them.');return}if(n==="off"){c.success("CLI autopilot is OFF. No new suggestions will be queued.");return}l&&u==="cli_queue"?c.success("CLI autopilot: ON (queueing fixes for your coding agent)."):l?c.info(`CLI autopilot: ON (delivery mode: ${u??"unknown"}).`):(c.info("CLI autopilot: OFF."),c.message("Run `seoagent autopilot on` to enable (paid)."))}import{existsSync as Tr,readFileSync as Or}from"fs";import{join as jr}from"path";var tn=["openai","fal","replicate"],F={openai:["OPENAI_API_KEY"],fal:["FAL_KEY","FAL_API_KEY"],replicate:["REPLICATE_API_TOKEN","REPLICATE_API_KEY"]};function Nr(e){let t={};if(!Tr(e))return t;let n=Or(e,"utf-8");for(let o of n.split(/\r?\n/)){let r=o.trim();if(!r||r.startsWith("#"))continue;let i=r.indexOf("=");if(i===-1)continue;let s=r.slice(0,i).trim(),a=r.slice(i+1).trim();(a.startsWith('"')&&a.endsWith('"')||a.startsWith("'")&&a.endsWith("'"))&&(a=a.slice(1,-1)),t[s]=a}return t}function nn(e,t){let n={};for(let o of t){let r=process.env[o];r&&r.trim().length>0&&(n[o]={value:r,source:"process"})}for(let o of A){let r=jr(e,o),i=Nr(r);for(let s of t){if(n[s])continue;let a=i[s];a&&a.length>0&&(n[s]={value:a,source:o})}}return{found:n}}function se(e){let t=[].concat(...tn.map(a=>F[a])),n=nn(e,t),o=[];for(let a of tn)F[a].some(l=>n.found[l])&&o.push(a);if(o.length===0)return{provider:"none",matched_key:null,source:null,available_providers:[]};let r=o[0],i=F[r].find(a=>n.found[a])??null,s=i?n.found[i].source:null;return{provider:r,matched_key:i,source:s,available_providers:o}}function on(e,t){let n=nn(e,F[t]);for(let o of F[t])if(n.found[o])return{key:n.found[o].value,envName:o};return null}function rn(){return F}function Me(e={}){let t=process.cwd(),n=f(t),o=se(t);if(e.json){process.stdout.write(JSON.stringify(o,null,2)+`
|
|
233
|
+
`);return}if(c.message("SEOAgent environment check"),o.provider==="none"){c.info("Image provider: none detected."),c.info("To enable image generation in the free tier, set ONE of:");let r=rn();for(let i of Object.keys(r))c.info(` \u2022 ${i}: ${r[i].join(" or ")}`);c.info("Add to your shell, .env.local, or .env. Then re-run `npx @seoagent-official/seoagent env-check`.")}else c.success(`Image provider: ${o.provider} (${o.matched_key} from ${o.source}).`),o.available_providers.length>1&&c.info(`Other providers available: ${o.available_providers.filter(r=>r!==o.provider).join(", ")}.`);if(!n){c.warn("No SEOAgent project here. Run `npx @seoagent-official/seoagent init` first to persist the provider.");return}et(t,{image_provider:o.provider}),c.info(`Saved to .seoagent/project.md (image_provider: ${o.provider}).`)}import{existsSync as Mr,mkdirSync as Br,writeFileSync as Gr}from"fs";import{dirname as cn,isAbsolute as Wr,resolve as Hr}from"path";import{Buffer as sn}from"buffer";async function Be(e){let t=await fetch(e);if(!t.ok)throw new Error(`Image download failed: ${t.status} ${t.statusText}`);let n=await t.arrayBuffer();return sn.from(n)}async function Lr(e){let t=e.size??"1024x1024",n=await fetch("https://api.openai.com/v1/images/generations",{method:"POST",headers:{Authorization:`Bearer ${e.apiKey}`,"Content-Type":"application/json"},body:JSON.stringify({model:"gpt-image-1",prompt:e.prompt,size:t,n:1,response_format:"b64_json"})});if(!n.ok){let i=await n.text().catch(()=>"");throw new Error(`OpenAI image API ${n.status}: ${i.slice(0,300)}`)}let r=(await n.json()).data?.[0];if(!r)throw new Error("OpenAI image API returned no data");if(r.b64_json)return{bytes:sn.from(r.b64_json,"base64"),contentType:"image/png"};if(r.url)return{bytes:await Be(r.url),contentType:"image/png"};throw new Error("OpenAI image API returned neither b64_json nor url")}async function Fr(e){let t=process.env.FAL_MODEL||"fal-ai/flux/schnell",n=await fetch(`https://fal.run/${t}`,{method:"POST",headers:{Authorization:`Key ${e.apiKey}`,"Content-Type":"application/json"},body:JSON.stringify({prompt:e.prompt,image_size:"landscape_16_9",num_images:1})});if(!n.ok){let s=await n.text().catch(()=>"");throw new Error(`fal.ai API ${n.status}: ${s.slice(0,300)}`)}let r=(await n.json()).images?.[0];if(!r?.url)throw new Error("fal.ai API returned no image url");return{bytes:await Be(r.url),contentType:r.content_type??"image/png"}}async function Ur(e,t,n=9e4){let o=Date.now(),r=1e3;for(;Date.now()-o<n;){let i=await fetch(`https://api.replicate.com/v1/predictions/${e}`,{headers:{Authorization:`Token ${t}`}});if(!i.ok)throw new Error(`Replicate poll failed: ${i.status}`);let s=await i.json();if(s.status==="succeeded"||s.status==="failed"||s.status==="canceled")return s;await new Promise(a=>setTimeout(a,r)),r=Math.min(r*1.5,5e3)}throw new Error("Replicate prediction timed out")}async function Dr(e){let t=process.env.REPLICATE_MODEL||"black-forest-labs/flux-schnell",n=await fetch(`https://api.replicate.com/v1/models/${t}/predictions`,{method:"POST",headers:{Authorization:`Token ${e.apiKey}`,"Content-Type":"application/json",Prefer:"wait"},body:JSON.stringify({input:{prompt:e.prompt,aspect_ratio:"16:9"}})});if(!n.ok){let s=await n.text().catch(()=>"");throw new Error(`Replicate API ${n.status}: ${s.slice(0,300)}`)}let o=await n.json();if(o.status!=="succeeded"&&o.status!=="failed"&&(o=await Ur(o.id,e.apiKey)),o.status!=="succeeded")throw new Error(`Replicate prediction ${o.status}: ${o.error??""}`);let r=Array.isArray(o.output)?o.output[0]:o.output;if(!r)throw new Error("Replicate prediction returned no output");return{bytes:await Be(r),contentType:"image/png",providerImageId:o.id}}async function an(e,t){switch(e){case"openai":return Lr(t);case"fal":return Fr(t);case"replicate":return Dr(t);default:throw new Error(`Unknown image provider: ${e}`)}}function Kr(e){return e==="openai"||e==="fal"||e==="replicate"}async function Ge(e={}){let t=process.cwd();if(!e.prompt){c.warn('Missing --prompt. Example: npx @seoagent-official/seoagent generate-image --prompt "..." --out content/images/hero.png'),process.exitCode=1;return}if(!e.out){c.warn("Missing --out. Example: --out .seoagent/content/images/hero.png"),process.exitCode=1;return}let n=f(t),o=null;if(e.provider){if(!Kr(e.provider)){c.warn(`Invalid --provider "${e.provider}". Use one of: openai, fal, replicate.`),process.exitCode=1;return}o=e.provider}else n?.image_provider&&n.image_provider!=="none"?o=n.image_provider:o=se(t).provider;if(!o||o==="none"){c.warn("No image generation provider available. Set OPENAI_API_KEY, FAL_KEY, or REPLICATE_API_TOKEN, then run `npx @seoagent-official/seoagent env-check`."),process.exitCode=1;return}let r=on(t,o);if(!r){c.warn(`Provider "${o}" selected but no API key found. Set the env var, then re-run.`),process.exitCode=1;return}let i=Wr(e.out)?e.out:Hr(t,e.out);e.silent||c.info(`Generating image with ${o} (${r.envName})...`);try{let s=await an(o,{prompt:e.prompt,apiKey:r.key,size:e.size});Mr(cn(i))||Br(cn(i),{recursive:!0}),Gr(i,s.bytes),e.silent||c.success(`Wrote ${i} (${s.bytes.length} bytes, ${s.contentType}).`)}catch(s){c.warn(`Image generation failed: ${s.message}`),process.exitCode=1}}var y=new zr;y.name("seoagent").description("AI SEO agent for Claude Code").version(V);y.command("init").description("Initialize SEOAgent project \u2014 creates .seoagent/ and installs the skill file").option("-y, --yes","Non-interactive: use inferred/env/flag values only (requires domain if not inferable)").option("--domain <domain>","Website domain (non-interactive or override)").option("--site-type <type>","Site type: saas, service, product, content, etc.").action(e=>{Ee({yes:e.yes,domain:e.domain,siteType:e.siteType})});y.command("uninstall").description("Remove SEOAgent from this project \u2014 deletes .seoagent/, the skill bundle, and the sync hook").option("-y, --yes","Skip the confirmation prompt (also implied in a non-TTY shell)").option("--global","Also wipe ~/.config/seoagent (login + sync state, all projects)").action(e=>{Ie({yes:e.yes,global:e.global})});y.command("status").description("Show current SEO project state").action(Ae);y.command("login").description("Connect this CLI to your seoagent.com account (browser flow)").option("--api-base <url>","Override API base URL (for testing)").action(e=>{Re({apiBase:e.apiBase})});y.command("logout").description("Remove stored credentials for seoagent.com").action(Te);y.command("sync").description("Sync .seoagent/ with your dashboard \u2014 push local changes then pull cloud changes (no-op when not logged in)").option("--silent","Suppress output (used by the Claude Code hook)").option("--force","Re-send every artifact on push; take cloud on every pull conflict").option("--path <relpath>","Push only files matching this path suffix").option("--push-only","Skip the cloud \u2192 local pull pass").option("--pull-only","Skip the local \u2192 cloud push pass (same as `seoagent pull`)").action(e=>{H({silent:e.silent,force:e.force,path:e.path,pushOnly:e.pushOnly,pullOnly:e.pullOnly})});y.command("pull").description("Pull cloud changes into .seoagent/ (dashboard / autopilot / chat edits)").option("--silent","Suppress output").option("--force","Take the cloud version on every conflict (discards local edits)").option("--print <path>","Read-only: print the current cloud body of one artifact to stdout (writes nothing). For conflict diffs.").action(e=>{Fe({force:e.force,silent:e.silent,print:e.print})});y.command("ack <action_id>").description("Close out a pending action from .seoagent/inbox/ (server marks completed)").option("--failed","Mark as failed instead of completed").option("--reason <text>","Reason for failure (used with --failed)").action((e,t)=>{Ue(e,{failed:t.failed,reason:t.reason})});y.command("autopilot <action>").description("Turn CLI autopilot on|off or show status (paid; queues fixes for your agent)").option("--api-base <url>","Override API base URL (for testing)").action((e,t)=>{De(e,{apiBase:t.apiBase})});y.command("env-check").description("Detect which image generation provider is available (OPENAI / FAL / REPLICATE)").option("--json","Output detection result as JSON").action(e=>{Me({json:e.json})});y.command("generate-image").description("Generate an image via the detected (or explicit) provider").requiredOption("--prompt <text>","Image prompt").requiredOption("--out <path>","Output file path (relative to cwd or absolute)").option("--provider <name>","Force provider: openai | fal | replicate").option("--size <wxh>","Pixel size hint, e.g. 1024x1024").option("--silent","Suppress progress output").action(e=>{Ge({prompt:e.prompt,out:e.out,provider:e.provider,size:e.size,silent:e.silent})});y.command("upgrade").description("Open SEOAgent Cloud pricing page").action(Ce);y.parse();
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@seoagent-official/seoagent",
|
|
3
|
-
"version": "1.
|
|
3
|
+
"version": "1.24.0",
|
|
4
4
|
"description": "Scaffolder for Claude Code's SEOAgent skill. Run once: `npx -y @seoagent-official/seoagent init`. Sets up .seoagent/ for persistent audits, keyword strategy, content planning, and optimized writing. Not a runtime dependency.",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"bin": {
|
package/skills/seoagent.md
CHANGED
|
@@ -92,7 +92,12 @@ Offer **once per session per topic**; if declined, drop it and keep working. Nev
|
|
|
92
92
|
|
|
93
93
|
## Processing the SEOAgent Inbox
|
|
94
94
|
|
|
95
|
-
`seoagent sync` also pulls **pending actions** from the dashboard into `.seoagent/inbox/`. These are autonomous decisions the cloud has made that need a human (or you, the AI agent) to apply in the user's local repo.
|
|
95
|
+
`seoagent sync` also pulls **pending actions** from the dashboard into `.seoagent/inbox/`. These are autonomous decisions the cloud has made that need a human (or you, the AI agent) to apply in the user's local repo. The action types today:
|
|
96
|
+
|
|
97
|
+
- **`cli_prune_pending`** — auto-prune decided an underperforming article should be removed from the repo (destructive — confirm first).
|
|
98
|
+
- **`cli_technical_fix`** — autopilot found an open technical-SEO issue (meta, schema, canonical, internal linking, …) to fix in a page's source. Safe/reversible (edits an existing page).
|
|
99
|
+
- **`cli_new_content`** — autopilot found a content brief with no article written yet. Write + publish the article. Safe (new content).
|
|
100
|
+
- **`cli_content_update`** — autopilot flagged an existing page to revise (declining GSC clicks, low CTR, or stale/thin). Reversible (edits existing content).
|
|
96
101
|
|
|
97
102
|
**Whenever the user says "process the inbox", "handle pending actions", "what's in my inbox", or anything similar**, OR whenever you see `.seoagent/inbox/README.md` reports pending actions after a sync, do this:
|
|
98
103
|
|
|
@@ -115,10 +120,37 @@ Offer **once per session per topic**; if declined, drop it and keep working. Nev
|
|
|
115
120
|
npx @seoagent-official/seoagent ack <action_id> --failed --reason "kept; performs well off-search"
|
|
116
121
|
```
|
|
117
122
|
|
|
118
|
-
3.
|
|
119
|
-
|
|
123
|
+
3. For each `cli_technical_fix-<id>.md` file:
|
|
124
|
+
- `Read` it. The frontmatter has `action_id`, `issue` (`meta`|`schema`|`canonical`|`internal_link`|`other`), `severity`, and `page_url`. The body describes the recommended fix and how to apply it per issue type.
|
|
125
|
+
- **Find the page's source** that renders `page_url` — the route/template/markdown under `app/`, `pages/`, `src/`, or `content/`. Match by URL path.
|
|
126
|
+
- Apply the fix in the source (use `Edit`/`Write`): meta → title/description (or the framework's metadata API/frontmatter); schema → JSON-LD; canonical → `<link rel="canonical">`; internal_link → add relevant internal links. These are **safe/reversible** edits to an existing page, so you don't need the hard delete-confirmation prune requires — but still **show the user the diff** (confirm once per session, then proceed).
|
|
127
|
+
- Acknowledge it server-side:
|
|
128
|
+
|
|
129
|
+
```bash
|
|
130
|
+
npx @seoagent-official/seoagent ack <action_id>
|
|
131
|
+
```
|
|
132
|
+
|
|
133
|
+
- If you disagree or it's a false positive, decline it:
|
|
134
|
+
|
|
135
|
+
```bash
|
|
136
|
+
npx @seoagent-official/seoagent ack <action_id> --failed --reason "not applicable; ..."
|
|
137
|
+
```
|
|
138
|
+
|
|
139
|
+
4. For each `cli_new_content-<id>.md` file:
|
|
140
|
+
- `Read` it. The frontmatter has `action_id`, `brief_slug`, `primary_keyword`, `cluster`, and `priority`. The body points at the synced brief.
|
|
141
|
+
- **Read the full brief** under `.seoagent/` (briefs file or `strategy/` entry matching `brief_slug`) for the outline, word-count target, and internal-link plan.
|
|
142
|
+
- Write the article following the skill's **content-production protocol** (Phase 4 below), then publish it where this project's content lives (repo `content/` or the connected CMS — you are the publishing engine). Show the user the draft before publishing.
|
|
143
|
+
- Acknowledge it: `npx @seoagent-official/seoagent ack <action_id>` (or `--failed --reason "skipped; off-strategy"` to decline).
|
|
144
|
+
|
|
145
|
+
5. For each `cli_content_update-<id>.md` file:
|
|
146
|
+
- `Read` it. The frontmatter has `action_id`, `reason` (`declining_clicks`|`low_ctr`|`stale_thin`), and `page_url`; the body has the signals.
|
|
147
|
+
- **Find the page's source** for `page_url`. Apply the revision per `reason`: `declining_clicks` → refresh/expand the content; `low_ctr` → rewrite title + meta description; `stale_thin` → expand and update. Follow the skill's **rewrite/revise protocol**. Reversible edit — show the user the diff (confirm once per session, then proceed).
|
|
148
|
+
- Acknowledge it: `npx @seoagent-official/seoagent ack <action_id>` (or `--failed --reason "kept as-is; ..."` to decline).
|
|
149
|
+
|
|
150
|
+
6. After processing, run `npx @seoagent-official/seoagent sync` once more to clean stale inbox files and confirm everything is settled.
|
|
151
|
+
7. Report a summary to the user: how many actions you applied, how many you declined (and why).
|
|
120
152
|
|
|
121
|
-
**Never delete a file without explicit user confirmation on the first action of the session.** Auto-prune is conservative (requires <5 clicks in 90 days, zero inbound internal links, etc.) but it can still surprise the user. Show them what's about to go.
|
|
153
|
+
**Never delete a file without explicit user confirmation on the first action of the session.** Auto-prune is conservative (requires <5 clicks in 90 days, zero inbound internal links, etc.) but it can still surprise the user. Show them what's about to go. (Technical-fix actions edit an existing page rather than delete, so they only need a diff review, not a destructive-action confirmation.)
|
|
122
154
|
|
|
123
155
|
## Output Format — Always Use This
|
|
124
156
|
|