@barbapapazes/content-creation 0.16.1 → 0.16.3

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
@@ -11,6 +11,7 @@ A CLI tool to streamline multi-platform content creation by generating dated dir
11
11
  - Config-driven templates for each content type
12
12
  - Never overwrites existing files (skip if exists behavior)
13
13
  - Generate separate index files per content type
14
+ - List upcoming content from today onward with absolute file paths
14
15
  - Link images to LinkedIn post frontmatter
15
16
  - Create resource directories (articles, videos, audio)
16
17
 
@@ -89,6 +90,30 @@ Total posts: 15
89
90
  ...
90
91
  ```
91
92
 
93
+ ### List Upcoming Content
94
+
95
+ List all content scheduled for today or later:
96
+
97
+ ```bash
98
+ # List upcoming content in current directory
99
+ content-creation list-upcoming
100
+
101
+ # List upcoming content at a specific path
102
+ content-creation list-upcoming --path /path/to/content
103
+ ```
104
+
105
+ This command scans your `YYYY/MM/DD` directory structure, reads the content title from frontmatter, and prints each matching content item with its absolute file path so it is easy to click from the terminal.
106
+
107
+ Example output:
108
+
109
+ ```text
110
+ 2026-04-18 · LinkedIn · Shipping a better content workflow
111
+ /absolute/path/to/content/2026/04/18/linkedin.md
112
+
113
+ 2026-04-20 · X · A tiny automation tip
114
+ /absolute/path/to/content/2026/04/20/x.md
115
+ ```
116
+
92
117
  ### Link Images
93
118
 
94
119
  Scan recent LinkedIn posts and link images to their frontmatter:
package/dist/cli.mjs CHANGED
@@ -1,2 +1,2 @@
1
1
  #!/usr/bin/env node
2
- import e from"node:process";import{intro as t,isCancel as n,log as r,multiselect as i,outro as a,select as o,text as s}from"@clack/prompts";import{cac as c}from"cac";import{addDays as l,format as u,parse as d,setHours as f,setMinutes as p,setSeconds as m}from"date-fns";import{existsSync as h,mkdirSync as g,readFileSync as _,readdirSync as v,statSync as y,writeFileSync as b}from"node:fs";import{dirname as x,isAbsolute as ee,join as S,resolve as C}from"node:path";import w from"gray-matter";import{loadConfig as T}from"c12";var E=`0.16.1`;const D={linkedin:{mainFile:`linkedin.md`,additionalFiles:[`linkedin-script.md`]},x:{mainFile:`x.md`},youtube:{mainFile:`youtube-script.md`,additionalFiles:[`youtube-description.md`]},instagram:{mainFile:`instagram-script.md`,additionalFiles:[`instagram-description.md`]}},O={thematic:[],templatesDir:void 0,templates:{},scheduling:{}};function k(t,n){return t?ee(t)?t:C(n?x(n):e.cwd(),t):n?x(n):e.cwd()}function te(e,t){let n={};if(e?.footerPath){let r=C(t,e.footerPath);h(r)&&(n.footerPath=r)}return n}function A(e,t){let n={};if(e?.templatePath){let r=C(t,e.templatePath);h(r)&&(n.templatePath=r)}return n}async function j(){let{config:t,configFile:n}=await T({name:`content-creation`,defaults:O,globalRc:!0,dotenv:!0}),r=k(t.templatesDir,n),i={linkedin:te(t.templates?.linkedin,r),youtube:A(t.templates?.youtube,r),instagram:A(t.templates?.instagram,r)},a={automationEndpoint:t.scheduling?.automationEndpoint||e.env.AUTOMATION_ENDPOINT,cfAccessClientId:t.scheduling?.cfAccessClientId||e.env.CF_ACCESS_CLIENT_ID,cfAccessClientSecret:t.scheduling?.cfAccessClientSecret||e.env.CF_ACCESS_CLIENT_SECRET};return{thematic:t.thematic||[],templatesDir:t.templatesDir||``,templates:i,scheduling:a}}function M(e){return h(e)?_(e,`utf-8`):``}function N(t,n,i,a,o=e.cwd()){let s=C(S(o,u(t,`yyyy`),u(t,`MM`),u(t,`dd`)));h(s)||(g(s,{recursive:!0}),r.success(`Created directory: ${s}`));for(let e of n)P(e,s,i,a);return s}function P(e,t,n,i){let a=D[e],o=S(t,a.mainFile);if(h(o)?r.info(`${a.mainFile} already exists, skipping`):F(e,o,n,i),a.additionalFiles)for(let o of a.additionalFiles){let a=S(t,o);L(o,e,n)&&(h(a)?r.info(`${o} already exists, skipping`):I(e,a,o,i))}}function F(e,t,n,i){let a={title:n.title};e===`linkedin`&&(n.theme&&(a.theme=n.theme),n.hasVideo&&(a.video=!0),n.hasImages&&(a.images=[null]));let o=``;if(e===`linkedin`){let e=i.templates.linkedin?.footerPath;o=e?M(e):``}b(t,w.stringify(o,a),`utf-8`),r.success(`Created ${t}`)}function I(e,t,n,i){let a=``;if(n.endsWith(`-description.md`)){let t=i.templates[e]?.templatePath;t&&(a=M(t))}b(t,a,`utf-8`),r.success(`Created ${t}`)}function L(e,t,n){return t===`linkedin`&&e===`linkedin-script.md`?n.hasVideo||!1:!0}function R(e,t){let n=[],i=D[t].mainFile;try{let t=v(e).filter(t=>y(S(e,t)).isDirectory()&&/^\d{4}$/.test(t));for(let a of t){let t=S(e,a),o=v(t).filter(e=>y(S(t,e)).isDirectory()&&/^\d{2}$/.test(e));for(let e of o){let o=S(t,e),s=v(o).filter(e=>y(S(o,e)).isDirectory()&&/^\d{2}$/.test(e));for(let t of s){let s=S(S(o,t),i);if(h(s)){let o=`${a}-${e}-${t}`,c=new Date(o);try{let{data:r}=w(_(s,`utf-8`)),l=r.title||`Untitled (${o})`;n.push({path:s,date:c,title:l,relativePath:`${a}/${e}/${t}/${i}`})}catch(e){r.warn(`Error reading ${s}: ${e}`)}}}}}}catch(e){return r.error(`Error reading directories: ${e}`),[]}return n.sort((e,t)=>t.date.getTime()-e.date.getTime())}function z(e){switch(e){case`linkedin`:return`linkedin-posts.md`;case`x`:return`x-posts.md`;case`youtube`:return`youtube-videos.md`;case`instagram`:return`instagram-posts.md`}}function B(e){switch(e){case`linkedin`:return`LinkedIn Posts`;case`x`:return`X Posts`;case`youtube`:return`YouTube Videos`;case`instagram`:return`Instagram Posts`}}function V(e,t){let n=R(e,t);if(n.length===0){r.info(`No ${t} posts found`);return}let i=B(t),a=z(t),o=`# ${i}\n\n`;o+=`_Generated on ${new Date().toLocaleDateString()}_\n\n`,o+=`Total posts: ${n.length}\n\n`;for(let e of n){let t=e.date.toLocaleDateString(`en-US`,{year:`numeric`,month:`long`,day:`numeric`});o+=`- [${e.title}](${e.relativePath}) - _${t}_\n`}b(S(e,a),o,`utf-8`),r.success(`Created ${a} with ${n.length} posts`)}function H(t=e.cwd(),n){let r=n||[`linkedin`,`x`,`youtube`,`instagram`];for(let e of r)V(t,e)}function U(e){return e.toLowerCase().trim().normalize(`NFD`).replace(/[\u0300-\u036F]/g,``).replace(/[^a-z0-9\s-]/g,``).replace(/\s+/g,`-`).replace(/-+/g,`-`).replace(/^-+|-+$/g,``)||`untitled`}function W(t,n,i=e.cwd()){let a=U(t),o=S(C(S(i,`resources`)),a),s=`${n}.md`,c=S(o,s);if(h(c))return r.info(`${s} already exists at: ${c}`),o;g(o,{recursive:!0}),r.success(`Created directory: ${o}`);let l={title:t,url:``,date:``};return b(c,w.stringify(``,l),`utf-8`),r.success(`Created ${s} at: ${c}`),o}function G(t=e.cwd()){let n=[];try{let e=v(t).filter(e=>y(S(t,e)).isDirectory()&&/^\d{4}$/.test(e));for(let r of e){let e=S(t,r),i=v(e).filter(t=>y(S(e,t)).isDirectory()&&/^\d{2}$/.test(t));for(let t of i){let i=S(e,t),a=v(i).filter(e=>y(S(i,e)).isDirectory()&&/^\d{2}$/.test(e));for(let e of a){let a=S(i,e);if(h(S(a,`linkedin.md`))){let i=`${r}-${t}-${e}`;n.push({path:a,date:new Date(i),displayPath:`${r}/${t}/${e}`})}}}}}catch(e){return r.error(`Error reading directories: ${e}`),[]}return n.sort((e,t)=>t.date.getTime()-e.date.getTime()).slice(0,8)}function K(e){let t=[`.jpg`,`.jpeg`,`.png`],n=[];try{let r=v(e);for(let i of r)if(y(S(e,i)).isFile()){let e=i.toLowerCase().substring(i.lastIndexOf(`.`));t.includes(e)&&n.push(i)}}catch(e){r.error(`Error reading directory: ${e}`)}return n.sort()}function q(e,t){try{let{data:n,content:i}=w(_(e,`utf-8`));n.images=t.length>0?t:[null],b(e,w.stringify(i,n),`utf-8`),r.success(`Updated ${e} with ${t.length} image(s)`)}catch(e){r.error(`Error updating frontmatter: ${e}`)}}async function J(t){if(t.length===0)return r.warn(`No dated folders with linkedin.md found`),null;let i=await o({message:`Select a folder to link images:`,options:t.map(e=>({label:e.displayPath,value:e.path,hint:`Link images in ${e.displayPath}`}))});return n(i)&&(r.error(`Operation cancelled.`),e.exit(0)),t.find(e=>e.path===i)||null}async function Y(t=e.cwd()){let n=G(t);if(n.length===0){r.warn(`No dated folders with linkedin.md found`);return}let i=await J(n);if(!i)return;let a=K(i.path);if(a.length===0){r.info(`No images found in ${i.displayPath}`),q(S(i.path,`linkedin.md`),[]);return}r.info(`Found ${a.length} image(s): ${a.join(`, `)}`),q(S(i.path,`linkedin.md`),a)}async function X(t,n){let i=S(t,`x.md`),{data:a,content:o}=w(_(i,`utf-8`));a.scheduled===!0&&(r.warn(`This content is already scheduled. Skipping.`),e.exit(0)),(!n.scheduling.automationEndpoint||!n.scheduling.cfAccessClientId||!n.scheduling.cfAccessClientSecret)&&(r.error(`Scheduling configuration is missing. Please set AUTOMATION_ENDPOINT, CF_ACCESS_CLIENT_ID, and CF_ACCESS_CLIENT_SECRET in your .env file or config.`),e.exit(1));let s=t.split(`/`),c=s[s.length-1],l=s[s.length-2],h=d(`${s[s.length-3]}-${l}-${c}`,`yyyy-MM-dd`,new Date);h=f(h,11),h=p(h,0),h=m(h,0);let g=h.getTime(),v={content:o.trim(),scheduleAt:g};try{let t=await fetch(n.scheduling.automationEndpoint,{method:`POST`,headers:{"Content-Type":`application/json`,Accept:`application/json`,"CF-Access-Client-Id":n.scheduling.cfAccessClientId,"CF-Access-Client-Secret":n.scheduling.cfAccessClientSecret},body:JSON.stringify(v)});if(!t.ok){let n=await t.text();r.error(`Failed to schedule reminder: ${t.status} ${t.statusText}\n${n}`),e.exit(1)}let s=await t.json(),c={...a,scheduled:!0};b(i,w.stringify(o,c),`utf-8`);let l=new Date(s.scheduledAt);r.success(`✓ Reminder scheduled successfully!`),r.info(`Scheduled for: ${u(l,`yyyy-MM-dd HH:mm:ss`)} UTC`)}catch(t){r.error(`Failed to schedule reminder: ${t instanceof Error?t.message:String(t)}`),e.exit(1)}}async function Z(){let t=new Date,i=[];for(let e=0;e<7;e++){let n=l(t,e),r=u(n,`yyyy-MM-dd`),a=u(n,`EEEE`),o=r,s=`Create content for ${a}, ${r}`;e===0?(o=`Today - ${a} (${r})`,s=`Create content for today (${a})`):e===1?(o=`Tomorrow - ${a} (${r})`,s=`Create content for tomorrow (${a})`):(o=`In ${e} days - ${a} (${r})`,s=`Create content for ${a}, ${r}`),i.push({label:o,value:`day-${e}`,hint:s})}i.push({label:`Custom date`,value:`custom`,hint:`Enter a specific date`});let a=await o({message:`When do you want to create content?`,options:i});if(n(a)&&(r.error(`Operation cancelled.`),e.exit(0)),a!==`custom`)return l(t,Number.parseInt(a.split(`-`)[1]));let c=await s({message:`Enter date (YYYY-MM-DD):`,placeholder:u(t,`yyyy-MM-dd`),validate:e=>{if(!e)return`Date is required`;if(!/^\d{4}-\d{2}-\d{2}$/.test(e))return`Invalid date format. Please use YYYY-MM-DD`;try{let t=d(e,`yyyy-MM-dd`,new Date);if(Number.isNaN(t.getTime()))return`Invalid date`}catch{return`Invalid date`}}});return n(c)&&(r.error(`Operation cancelled.`),e.exit(0)),d(c,`yyyy-MM-dd`,new Date)}async function Q(){let t=await s({message:`What is the title of your content?`,placeholder:`Enter content title`,validate:e=>{if(!e||e.trim().length===0)return`Title is required`}});return n(t)&&(r.error(`Operation cancelled.`),e.exit(0)),t.trim()}async function ne(){let t=await o({message:`Is a LinkedIn video planned?`,options:[{label:`Yes`,value:!0,hint:`Add video: true to frontmatter`},{label:`No`,value:!1,hint:`No video metadata`}]});return n(t)&&(r.error(`Operation cancelled.`),e.exit(0)),t}async function re(){let t=await o({message:`Are there images?`,options:[{label:`Yes`,value:!0,hint:`Add images section to frontmatter`},{label:`No`,value:!1,hint:`No images metadata`}]});return n(t)&&(r.error(`Operation cancelled.`),e.exit(0)),t}async function ie(t){if(t.length===0)return;let i=await o({message:`What is the content theme? (optional)`,options:[{label:`No theme`,value:``,hint:`Leave theme empty`},...t.map(e=>({label:e,value:e,hint:`Use theme: ${e}`}))]});n(i)&&(r.error(`Operation cancelled.`),e.exit(0));let a=i;return a.length>0?a:void 0}async function ae(){let t=await o({message:`What type of resource is this?`,options:[{label:`Article`,value:`article`,hint:`Create article.md`},{label:`Video`,value:`video`,hint:`Create video.md`},{label:`Audio`,value:`audio`,hint:`Create audio.md`},{label:`Tweet`,value:`tweet`,hint:`Create tweet.md`}]});return n(t)&&(r.error(`Operation cancelled.`),e.exit(0)),t}async function oe(){let t=await i({message:`Which content types do you want to create?`,options:[{label:`LinkedIn`,value:`linkedin`,hint:`Create LinkedIn post and optional script`},{label:`X (Twitter)`,value:`x`,hint:`Create X post`},{label:`YouTube`,value:`youtube`,hint:`Create YouTube script and description`},{label:`Instagram`,value:`instagram`,hint:`Create Instagram script and description`}],required:!0});return n(t)&&(r.error(`Operation cancelled.`),e.exit(0)),t}async function se(t){let i=[];h(t)||(r.error(`Base path does not exist: ${t}`),e.exit(1));let a=v(t).filter(e=>y(S(t,e)).isDirectory()&&/^\d{4}$/.test(e));for(let e of a){let n=S(t,e),r=v(n).filter(e=>y(S(n,e)).isDirectory()&&/^\d{2}$/.test(e));for(let t of r){let r=S(n,t),a=v(r).filter(e=>y(S(r,e)).isDirectory()&&/^\d{2}$/.test(e));for(let n of a){let a=S(r,n);if(h(S(a,`x.md`))){let r=`${e}-${t}-${n}`;i.push({path:a,date:r,title:``})}}}}i.length===0&&(r.error(`No x.md files found in dated folders`),e.exit(1)),i.sort((e,t)=>t.date.localeCompare(e.date));let s=await o({message:`Which content do you want to schedule a reminder for?`,options:i.map(e=>({label:e.date,value:e.path,hint:`Schedule reminder for ${e.date}`}))});return n(s)&&(r.error(`Operation cancelled.`),e.exit(0)),s}const $=c(`content-creation`);$.command(`[path]`,`Create a dated content directory with content files`).option(`--path <path>`,`Base path for content directory (defaults to current directory)`).action(async(n,r)=>{t(`Content Creation - Create dated content directory`);let i=await j(),o=await Q(),s=await oe(),c={title:o};s.includes(`linkedin`)&&(c.hasVideo=await ne(),c.hasImages=await re(),i.thematic.length>0&&(c.theme=await ie(i.thematic)));let l=await Z();N(l,s,c,i,n||r?.path||e.cwd());let d=s.join(`, `);a(`✓ Content directory created: ${u(l,`yyyy/MM/dd`)} (${d})`)}),$.command(`link-images [path]`,`Link images to LinkedIn frontmatter in recent folders`).option(`--path <path>`,`Base path for content directory (defaults to current directory)`).action(async(n,r)=>{t(`Content Creation - Link Images to LinkedIn`),await Y(n||r?.path||e.cwd()),a(`✓ Images linked successfully`)}),$.command(`resource [path]`,`Create a resource with article/video/audio markdown file`).option(`--path <path>`,`Base path for resource directory (defaults to current directory)`).action(async(n,r)=>{t(`Content Creation - Create Resource`);let i=await Q(),o=await ae();W(i,o,n||r?.path||e.cwd()),a(`✓ Resource created: resources/${U(i)}/${o}.md`)}),$.command(`create-index [path]`,`Create index files for each content type`).option(`--path <path>`,`Base path for content directory (defaults to current directory)`).action(async(n,r)=>{t(`Content Creation - Create Index Files`),H(n||r?.path||e.cwd()),a(`✓ Index files generated successfully`)}),$.command(`schedule-reminder [path]`,`Schedule a reminder for X content`).option(`--path <path>`,`Base path for content directory (defaults to current directory)`).action(async(n,r)=>{t(`Content Creation - Schedule X Reminder`);let i=await j();await X(await se(n||r?.path||e.cwd()),i),a(`✓ Reminder scheduled successfully`)}),$.help(),$.version(E),$.parse();export{};
2
+ import e from"node:process";import{intro as t,isCancel as n,log as r,multiselect as i,outro as a,select as o,text as s}from"@clack/prompts";import{cac as c}from"cac";import{addDays as l,format as u,parse as d,setHours as f,setMinutes as ee,setSeconds as p}from"date-fns";import{existsSync as m,mkdirSync as h,readFileSync as g,readdirSync as _,statSync as v,writeFileSync as y}from"node:fs";import{dirname as b,isAbsolute as x,join as S,resolve as C}from"node:path";import w from"gray-matter";import{loadConfig as T}from"c12";var E=`0.16.3`;const D={linkedin:{mainFile:`linkedin.md`,additionalFiles:[`linkedin-script.md`]},x:{mainFile:`x.md`},youtube:{mainFile:`youtube-script.md`,additionalFiles:[`youtube-description.md`]},instagram:{mainFile:`instagram-script.md`,additionalFiles:[`instagram-description.md`]}},O={thematic:[],templatesDir:void 0,templates:{},scheduling:{}};function te(t,n){return t?x(t)?t:C(n?b(n):e.cwd(),t):n?b(n):e.cwd()}function ne(e,t){let n={};if(e?.footerPath){let r=C(t,e.footerPath);m(r)&&(n.footerPath=r)}return n}function k(e,t){let n={};if(e?.templatePath){let r=C(t,e.templatePath);m(r)&&(n.templatePath=r)}return n}async function A(){let{config:t,configFile:n}=await T({name:`content-creation`,defaults:O,globalRc:!0,dotenv:!0}),r=te(t.templatesDir,n),i={linkedin:ne(t.templates?.linkedin,r),youtube:k(t.templates?.youtube,r),instagram:k(t.templates?.instagram,r)},a={automationEndpoint:t.scheduling?.automationEndpoint||e.env.AUTOMATION_ENDPOINT,cfAccessClientId:t.scheduling?.cfAccessClientId||e.env.CF_ACCESS_CLIENT_ID,cfAccessClientSecret:t.scheduling?.cfAccessClientSecret||e.env.CF_ACCESS_CLIENT_SECRET};return{thematic:t.thematic||[],templatesDir:t.templatesDir||``,templates:i,scheduling:a}}function j(e){return m(e)?g(e,`utf-8`):``}function M(t,n,i,a,o=e.cwd()){let s=C(S(o,u(t,`yyyy`),u(t,`MM`),u(t,`dd`)));m(s)||(h(s,{recursive:!0}),r.success(`Created directory: ${s}`));for(let e of n)N(e,s,i,a);return s}function N(e,t,n,i){let a=D[e],o=S(t,a.mainFile);if(m(o)?r.info(`${a.mainFile} already exists, skipping`):P(e,o,n,i),a.additionalFiles)for(let o of a.additionalFiles){let a=S(t,o);I(o,e,n)&&(m(a)?r.info(`${o} already exists, skipping`):F(e,a,o,i))}}function P(e,t,n,i){let a={title:n.title};e===`linkedin`&&(n.theme&&(a.theme=n.theme),n.hasVideo&&(a.video=!0),n.hasImages&&(a.images=[null]));let o=``;if(e===`linkedin`){let e=i.templates.linkedin?.footerPath;o=e?j(e):``}y(t,w.stringify(o,a),`utf-8`),r.success(`Created ${t}`)}function F(e,t,n,i){let a=``;if(n.endsWith(`-description.md`)){let t=i.templates[e]?.templatePath;t&&(a=j(t))}y(t,a,`utf-8`),r.success(`Created ${t}`)}function I(e,t,n){return t===`linkedin`&&e===`linkedin-script.md`?n.hasVideo||!1:!0}const L=[`linkedin`,`x`,`youtube`,`instagram`];function R(e,t){let n=[],i=D[t].mainFile;try{let t=_(e).filter(t=>v(S(e,t)).isDirectory()&&/^\d{4}$/.test(t));for(let a of t){let t=S(e,a),o=_(t).filter(e=>v(S(t,e)).isDirectory()&&/^\d{2}$/.test(e));for(let e of o){let o=S(t,e),s=_(o).filter(e=>v(S(o,e)).isDirectory()&&/^\d{2}$/.test(e));for(let t of s){let s=C(S(o,t),i);if(m(s)){let o=`${a}-${e}-${t}`,c=new Date(Number(a),Number(e)-1,Number(t));try{let{data:r}=w(g(s,`utf-8`)),l=r.title||`Untitled (${o})`;n.push({path:s,date:c,title:l,relativePath:`${a}/${e}/${t}/${i}`})}catch(e){r.warn(`Error reading ${s}: ${e}`)}}}}}}catch(e){return r.error(`Error reading directories: ${e}`),[]}return n.sort((e,t)=>t.date.getTime()-e.date.getTime())}function z(e){switch(e){case`linkedin`:return`linkedin-posts.md`;case`x`:return`x-posts.md`;case`youtube`:return`youtube-videos.md`;case`instagram`:return`instagram-posts.md`}}function B(e){switch(e){case`linkedin`:return`LinkedIn Posts`;case`x`:return`X Posts`;case`youtube`:return`YouTube Videos`;case`instagram`:return`Instagram Posts`}}function V(e,t){let n=R(e,t);if(n.length===0){r.info(`No ${t} posts found`);return}let i=B(t),a=z(t),o=`# ${i}\n\n`;o+=`_Generated on ${new Date().toLocaleDateString()}_\n\n`,o+=`Total posts: ${n.length}\n\n`;for(let e of n){let t=e.date.toLocaleDateString(`en-US`,{year:`numeric`,month:`long`,day:`numeric`});o+=`- [${e.title}](${e.relativePath}) - _${t}_\n`}y(S(e,a),o,`utf-8`),r.success(`Created ${a} with ${n.length} posts`)}function H(t=e.cwd(),n){let r=n||[...L];for(let e of r)V(t,e)}function U(e){return e.toLowerCase().trim().normalize(`NFD`).replace(/[\u0300-\u036F]/g,``).replace(/[^a-z0-9\s-]/g,``).replace(/\s+/g,`-`).replace(/-+/g,`-`).replace(/^-+|-+$/g,``)||`untitled`}function W(t,n,i=e.cwd()){let a=U(t),o=S(C(S(i,`resources`)),a),s=`${n}.md`,c=S(o,s);if(m(c))return r.info(`${s} already exists at: ${c}`),o;h(o,{recursive:!0}),r.success(`Created directory: ${o}`);let l={title:t,url:``,date:``};return y(c,w.stringify(``,l),`utf-8`),r.success(`Created ${s} at: ${c}`),o}function G(t=e.cwd()){let n=[];try{let e=_(t).filter(e=>v(S(t,e)).isDirectory()&&/^\d{4}$/.test(e));for(let r of e){let e=S(t,r),i=_(e).filter(t=>v(S(e,t)).isDirectory()&&/^\d{2}$/.test(t));for(let t of i){let i=S(e,t),a=_(i).filter(e=>v(S(i,e)).isDirectory()&&/^\d{2}$/.test(e));for(let e of a){let a=S(i,e);if(m(S(a,`linkedin.md`))){let i=`${r}-${t}-${e}`;n.push({path:a,date:new Date(i),displayPath:`${r}/${t}/${e}`})}}}}}catch(e){return r.error(`Error reading directories: ${e}`),[]}return n.sort((e,t)=>t.date.getTime()-e.date.getTime()).slice(0,8)}function K(e){let t=[`.jpg`,`.jpeg`,`.png`],n=[];try{let r=_(e);for(let i of r)if(v(S(e,i)).isFile()){let e=i.toLowerCase().substring(i.lastIndexOf(`.`));t.includes(e)&&n.push(i)}}catch(e){r.error(`Error reading directory: ${e}`)}return n.sort()}function q(e,t){try{let{data:n,content:i}=w(g(e,`utf-8`));n.images=t.length>0?t:[null],y(e,w.stringify(i,n),`utf-8`),r.success(`Updated ${e} with ${t.length} image(s)`)}catch(e){r.error(`Error updating frontmatter: ${e}`)}}async function J(t){if(t.length===0)return r.warn(`No dated folders with linkedin.md found`),null;let i=await o({message:`Select a folder to link images:`,options:t.map(e=>({label:e.displayPath,value:e.path,hint:`Link images in ${e.displayPath}`}))});return n(i)&&(r.error(`Operation cancelled.`),e.exit(0)),t.find(e=>e.path===i)||null}async function Y(t=e.cwd()){let n=G(t);if(n.length===0){r.warn(`No dated folders with linkedin.md found`);return}let i=await J(n);if(!i)return;let a=K(i.path);if(a.length===0){r.info(`No images found in ${i.displayPath}`),q(S(i.path,`linkedin.md`),[]);return}r.info(`Found ${a.length} image(s): ${a.join(`, `)}`),q(S(i.path,`linkedin.md`),a)}const X={linkedin:`LinkedIn`,x:`X`,youtube:`YouTube`,instagram:`Instagram`};function re(e){let t=new Date;t.setHours(0,0,0,0);let n=L.flatMap(t=>R(e,t).map(e=>({...e,contentType:t}))).filter(e=>e.date.getTime()>=t.getTime()).sort((e,t)=>{let n=e.date.getTime()-t.date.getTime();return n===0?L.indexOf(e.contentType)-L.indexOf(t.contentType):n});if(n.length===0){r.info(`No upcoming content found from today onward.`);return}r.info(`Found ${n.length} upcoming content item${n.length===1?``:`s`}:`),console.log(``);for(let e of n)console.log(`${u(e.date,`yyyy-MM-dd`)} · ${X[e.contentType]} · ${e.title}`),console.log(e.path),console.log(``)}async function ie(t,n){let i=S(t,`x.md`),{data:a,content:o}=w(g(i,`utf-8`));a.scheduled===!0&&(r.warn(`This content is already scheduled. Skipping.`),e.exit(0)),(!n.scheduling.automationEndpoint||!n.scheduling.cfAccessClientId||!n.scheduling.cfAccessClientSecret)&&(r.error(`Scheduling configuration is missing. Please set AUTOMATION_ENDPOINT, CF_ACCESS_CLIENT_ID, and CF_ACCESS_CLIENT_SECRET in your .env file or config.`),e.exit(1));let s=t.split(`/`),c=s[s.length-1],l=s[s.length-2],m=d(`${s[s.length-3]}-${l}-${c}`,`yyyy-MM-dd`,new Date);m=f(m,11),m=ee(m,0),m=p(m,0);let h=m.getTime(),_={content:o.trim(),scheduleAt:h};try{let t=await fetch(n.scheduling.automationEndpoint,{method:`POST`,headers:{"Content-Type":`application/json`,Accept:`application/json`,"CF-Access-Client-Id":n.scheduling.cfAccessClientId,"CF-Access-Client-Secret":n.scheduling.cfAccessClientSecret},body:JSON.stringify(_)});if(!t.ok){let n=await t.text();r.error(`Failed to schedule reminder: ${t.status} ${t.statusText}\n${n}`),e.exit(1)}let s=await t.json(),c={...a,scheduled:!0};y(i,w.stringify(o,c),`utf-8`);let l=new Date(s.scheduledAt);r.success(`✓ Reminder scheduled successfully!`),r.info(`Scheduled for: ${u(l,`yyyy-MM-dd HH:mm:ss`)} UTC`)}catch(t){r.error(`Failed to schedule reminder: ${t instanceof Error?t.message:String(t)}`),e.exit(1)}}async function ae(){let t=new Date,i=[];for(let e=0;e<7;e++){let n=l(t,e),r=u(n,`yyyy-MM-dd`),a=u(n,`EEEE`),o=r,s=`Create content for ${a}, ${r}`;e===0?(o=`Today - ${a} (${r})`,s=`Create content for today (${a})`):e===1?(o=`Tomorrow - ${a} (${r})`,s=`Create content for tomorrow (${a})`):(o=`In ${e} days - ${a} (${r})`,s=`Create content for ${a}, ${r}`),i.push({label:o,value:`day-${e}`,hint:s})}i.push({label:`Custom date`,value:`custom`,hint:`Enter a specific date`});let a=await o({message:`When do you want to create content?`,options:i});if(n(a)&&(r.error(`Operation cancelled.`),e.exit(0)),a!==`custom`)return l(t,Number.parseInt(a.split(`-`)[1]));let c=await s({message:`Enter date (YYYY-MM-DD):`,placeholder:u(t,`yyyy-MM-dd`),validate:e=>{if(!e)return`Date is required`;if(!/^\d{4}-\d{2}-\d{2}$/.test(e))return`Invalid date format. Please use YYYY-MM-DD`;try{let t=d(e,`yyyy-MM-dd`,new Date);if(Number.isNaN(t.getTime()))return`Invalid date`}catch{return`Invalid date`}}});return n(c)&&(r.error(`Operation cancelled.`),e.exit(0)),d(c,`yyyy-MM-dd`,new Date)}async function Z(){let t=await s({message:`What is the title of your content?`,placeholder:`Enter content title`,validate:e=>{if(!e||e.trim().length===0)return`Title is required`}});return n(t)&&(r.error(`Operation cancelled.`),e.exit(0)),t.trim()}async function oe(){let t=await o({message:`Is a LinkedIn video planned?`,options:[{label:`Yes`,value:!0,hint:`Add video: true to frontmatter`},{label:`No`,value:!1,hint:`No video metadata`}]});return n(t)&&(r.error(`Operation cancelled.`),e.exit(0)),t}async function se(){let t=await o({message:`Are there images?`,options:[{label:`Yes`,value:!0,hint:`Add images section to frontmatter`},{label:`No`,value:!1,hint:`No images metadata`}]});return n(t)&&(r.error(`Operation cancelled.`),e.exit(0)),t}async function Q(t){if(t.length===0)return;let i=await o({message:`What is the content theme? (optional)`,options:[{label:`No theme`,value:``,hint:`Leave theme empty`},...t.map(e=>({label:e,value:e,hint:`Use theme: ${e}`}))]});n(i)&&(r.error(`Operation cancelled.`),e.exit(0));let a=i;return a.length>0?a:void 0}async function ce(){let t=await o({message:`What type of resource is this?`,options:[{label:`Article`,value:`article`,hint:`Create article.md`},{label:`Video`,value:`video`,hint:`Create video.md`},{label:`Audio`,value:`audio`,hint:`Create audio.md`},{label:`Tweet`,value:`tweet`,hint:`Create tweet.md`}]});return n(t)&&(r.error(`Operation cancelled.`),e.exit(0)),t}async function le(){let t=await i({message:`Which content types do you want to create?`,options:[{label:`LinkedIn`,value:`linkedin`,hint:`Create LinkedIn post and optional script`},{label:`X (Twitter)`,value:`x`,hint:`Create X post`},{label:`YouTube`,value:`youtube`,hint:`Create YouTube script and description`},{label:`Instagram`,value:`instagram`,hint:`Create Instagram script and description`}],required:!0});return n(t)&&(r.error(`Operation cancelled.`),e.exit(0)),t}async function ue(t){let i=[];m(t)||(r.error(`Base path does not exist: ${t}`),e.exit(1));let a=_(t).filter(e=>v(S(t,e)).isDirectory()&&/^\d{4}$/.test(e));for(let e of a){let n=S(t,e),r=_(n).filter(e=>v(S(n,e)).isDirectory()&&/^\d{2}$/.test(e));for(let t of r){let r=S(n,t),a=_(r).filter(e=>v(S(r,e)).isDirectory()&&/^\d{2}$/.test(e));for(let n of a){let a=S(r,n);if(m(S(a,`x.md`))){let r=`${e}-${t}-${n}`;i.push({path:a,date:r,title:``})}}}}i.length===0&&(r.error(`No x.md files found in dated folders`),e.exit(1)),i.sort((e,t)=>t.date.localeCompare(e.date));let s=await o({message:`Which content do you want to schedule a reminder for?`,options:i.map(e=>({label:e.date,value:e.path,hint:`Schedule reminder for ${e.date}`}))});return n(s)&&(r.error(`Operation cancelled.`),e.exit(0)),s}const $=c(`content-creation`);$.command(`[path]`,`Create a dated content directory with content files`).option(`--path <path>`,`Base path for content directory (defaults to current directory)`).action(async(n,r)=>{t(`Content Creation - Create dated content directory`);let i=await A(),o=await Z(),s=await le(),c={title:o};s.includes(`linkedin`)&&(c.hasVideo=await oe(),c.hasImages=await se(),i.thematic.length>0&&(c.theme=await Q(i.thematic)));let l=await ae();M(l,s,c,i,n||r?.path||e.cwd());let d=s.join(`, `);a(`✓ Content directory created: ${u(l,`yyyy/MM/dd`)} (${d})`)}),$.command(`link-images [path]`,`Link images to LinkedIn frontmatter in recent folders`).option(`--path <path>`,`Base path for content directory (defaults to current directory)`).action(async(n,r)=>{t(`Content Creation - Link Images to LinkedIn`),await Y(n||r?.path||e.cwd()),a(`✓ Images linked successfully`)}),$.command(`resource [path]`,`Create a resource with article/video/audio markdown file`).option(`--path <path>`,`Base path for resource directory (defaults to current directory)`).action(async(n,r)=>{t(`Content Creation - Create Resource`);let i=await Z(),o=await ce();W(i,o,n||r?.path||e.cwd()),a(`✓ Resource created: resources/${U(i)}/${o}.md`)}),$.command(`create-index [path]`,`Create index files for each content type`).option(`--path <path>`,`Base path for content directory (defaults to current directory)`).action(async(n,r)=>{t(`Content Creation - Create Index Files`),H(n||r?.path||e.cwd()),a(`✓ Index files generated successfully`)}),$.command(`list-upcoming [path]`,`List content scheduled for today or later`).option(`--path <path>`,`Base path for content directory (defaults to current directory)`).action(async(n,r)=>{t(`Content Creation - List Upcoming Content`),re(n||r?.path||e.cwd()),a(`✓ Upcoming content listed successfully`)}),$.command(`schedule-reminder [path]`,`Schedule a reminder for X content`).option(`--path <path>`,`Base path for content directory (defaults to current directory)`).action(async(n,r)=>{t(`Content Creation - Schedule X Reminder`);let i=await A();await ie(await ue(n||r?.path||e.cwd()),i),a(`✓ Reminder scheduled successfully`)}),$.help(),$.version(E),$.parse();export{};
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@barbapapazes/content-creation",
3
3
  "type": "module",
4
- "version": "0.16.1",
4
+ "version": "0.16.3",
5
5
  "author": "Estéban Soubiran <esteban@soubiran.dev>",
6
6
  "license": "MIT",
7
7
  "funding": "https://github.com/sponsors/Barbapapazes",
@@ -31,15 +31,15 @@
31
31
  },
32
32
  "dependencies": {
33
33
  "@clack/prompts": "^0.10.1",
34
- "c12": "^3.3.3",
34
+ "c12": "^3.3.4",
35
35
  "cac": "^6.7.14",
36
36
  "date-fns": "^4.1.0",
37
37
  "gray-matter": "^4.0.3"
38
38
  },
39
39
  "devDependencies": {
40
- "@tsconfig/node24": "^24.0.3",
41
- "@types/node": "^22.19.3",
42
- "tsdown": "^0.18.3",
40
+ "@tsconfig/node24": "^24.0.4",
41
+ "@types/node": "^22.19.17",
42
+ "tsdown": "^0.18.4",
43
43
  "typescript": "^5.9.3"
44
44
  },
45
45
  "scripts": {