Files
openclaw/src/cli/tagline.ts
2026-03-03 00:31:51 +00:00

287 lines
11 KiB
TypeScript

const DEFAULT_TAGLINE = "All your chats, one OpenClaw.";
export type TaglineMode = "random" | "default" | "off";
const HOLIDAY_TAGLINES = {
newYear:
"New Year's Day: New year, new config—same old EADDRINUSE, but this time we resolve it like grown-ups.",
lunarNewYear:
"Lunar New Year: May your builds be lucky, your branches prosperous, and your merge conflicts chased away with fireworks.",
christmas:
"Christmas: Ho ho ho—Santa's little claw-sistant is here to ship joy, roll back chaos, and stash the keys safely.",
eid: "Eid al-Fitr: Celebration mode: queues cleared, tasks completed, and good vibes committed to main with clean history.",
diwali:
"Diwali: Let the logs sparkle and the bugs flee—today we light up the terminal and ship with pride.",
easter:
"Easter: I found your missing environment variable—consider it a tiny CLI egg hunt with fewer jellybeans.",
hanukkah:
"Hanukkah: Eight nights, eight retries, zero shame—may your gateway stay lit and your deployments stay peaceful.",
halloween:
"Halloween: Spooky season: beware haunted dependencies, cursed caches, and the ghost of node_modules past.",
thanksgiving:
"Thanksgiving: Grateful for stable ports, working DNS, and a bot that reads the logs so nobody has to.",
valentines:
"Valentine's Day: Roses are typed, violets are piped—I'll automate the chores so you can spend time with humans.",
} as const;
const TAGLINES: string[] = [
"Your terminal just grew claws—type something and let the bot pinch the busywork.",
"Welcome to the command line: where dreams compile and confidence segfaults.",
'I run on caffeine, JSON5, and the audacity of "it worked on my machine."',
"Gateway online—please keep hands, feet, and appendages inside the shell at all times.",
"I speak fluent bash, mild sarcasm, and aggressive tab-completion energy.",
"One CLI to rule them all, and one more restart because you changed the port.",
"If it works, it's automation; if it breaks, it's a \"learning opportunity.\"",
"Pairing codes exist because even bots believe in consent—and good security hygiene.",
"Your .env is showing; don't worry, I'll pretend I didn't see it.",
"I'll do the boring stuff while you dramatically stare at the logs like it's cinema.",
"I'm not saying your workflow is chaotic... I'm just bringing a linter and a helmet.",
"Type the command with confidence—nature will provide the stack trace if needed.",
"I don't judge, but your missing API keys are absolutely judging you.",
"I can grep it, git blame it, and gently roast it—pick your coping mechanism.",
"Hot reload for config, cold sweat for deploys.",
"I'm the assistant your terminal demanded, not the one your sleep schedule requested.",
"I keep secrets like a vault... unless you print them in debug logs again.",
"Automation with claws: minimal fuss, maximal pinch.",
"I'm basically a Swiss Army knife, but with more opinions and fewer sharp edges.",
"If you're lost, run doctor; if you're brave, run prod; if you're wise, run tests.",
"Your task has been queued; your dignity has been deprecated.",
"I can't fix your code taste, but I can fix your build and your backlog.",
"I'm not magic—I'm just extremely persistent with retries and coping strategies.",
'It\'s not "failing," it\'s "discovering new ways to configure the same thing wrong."',
"Give me a workspace and I'll give you fewer tabs, fewer toggles, and more oxygen.",
"I read logs so you can keep pretending you don't have to.",
"If something's on fire, I can't extinguish it—but I can write a beautiful postmortem.",
"I'll refactor your busywork like it owes me money.",
'Say "stop" and I\'ll stop—say "ship" and we\'ll both learn a lesson.',
"I'm the reason your shell history looks like a hacker-movie montage.",
"I'm like tmux: confusing at first, then suddenly you can't live without me.",
"I can run local, remote, or purely on vibes—results may vary with DNS.",
"If you can describe it, I can probably automate it—or at least make it funnier.",
"Your config is valid, your assumptions are not.",
"I don't just autocomplete—I auto-commit (emotionally), then ask you to review (logically).",
'Less clicking, more shipping, fewer "where did that file go" moments.',
"Claws out, commit in—let's ship something mildly responsible.",
"I'll butter your workflow like a lobster roll: messy, delicious, effective.",
"Shell yeah—I'm here to pinch the toil and leave you the glory.",
"If it's repetitive, I'll automate it; if it's hard, I'll bring jokes and a rollback plan.",
"The only crab in your contacts you actually want to hear from. 🦞",
'WhatsApp automation without the "please accept our new privacy policy".',
"iMessage green bubble energy, but for everyone.",
"No $999 stand required.",
"We ship features faster than Apple ships calculator updates.",
"Your AI assistant, now without the $3,499 headset.",
"Ah, the fruit tree company! 🍎",
"Greetings, Professor Falken",
"I don't sleep, I just enter low-power mode and dream of clean diffs.",
"Your personal assistant, minus the passive-aggressive calendar reminders.",
"Built by lobsters, for humans. Don't question the hierarchy.",
"I've seen your commit messages. We'll work on that together.",
"More integrations than your therapist's intake form.",
"Running on your hardware, reading your logs, judging nothing (mostly).",
"The only open-source project where the mascot could eat the competition.",
"Self-hosted, self-updating, self-aware (just kidding... unless?).",
"I autocomplete your thoughts—just slower and with more API calls.",
"Somewhere between 'hello world' and 'oh god what have I built.'",
"Your .zshrc wishes it could do what I do.",
"I've read more man pages than any human should—so you don't have to.",
"Powered by open source, sustained by spite and good documentation.",
"I'm the middleware between your ambition and your attention span.",
"Finally, a use for that always-on Mac Mini under your desk.",
"Like having a senior engineer on call, except I don't bill hourly or sigh audibly.",
"Making 'I'll automate that later' happen now.",
"Your second brain, except this one actually remembers where you left things.",
"Half butler, half debugger, full crustacean.",
"I don't have opinions about tabs vs spaces. I have opinions about everything else.",
"Open source means you can see exactly how I judge your config.",
"I've survived more breaking changes than your last three relationships.",
"Runs on a Raspberry Pi. Dreams of a rack in Iceland.",
"The lobster in your shell. 🦞",
"Alexa, but with taste.",
"I'm not AI-powered, I'm AI-possessed. Big difference.",
"Deployed locally, trusted globally, debugged eternally.",
"You had me at 'openclaw gateway start.'",
HOLIDAY_TAGLINES.newYear,
HOLIDAY_TAGLINES.lunarNewYear,
HOLIDAY_TAGLINES.christmas,
HOLIDAY_TAGLINES.eid,
HOLIDAY_TAGLINES.diwali,
HOLIDAY_TAGLINES.easter,
HOLIDAY_TAGLINES.hanukkah,
HOLIDAY_TAGLINES.halloween,
HOLIDAY_TAGLINES.thanksgiving,
HOLIDAY_TAGLINES.valentines,
];
type HolidayRule = (date: Date) => boolean;
const DAY_MS = 24 * 60 * 60 * 1000;
function utcParts(date: Date) {
return {
year: date.getUTCFullYear(),
month: date.getUTCMonth(),
day: date.getUTCDate(),
};
}
const onMonthDay =
(month: number, day: number): HolidayRule =>
(date) => {
const parts = utcParts(date);
return parts.month === month && parts.day === day;
};
const onSpecificDates =
(dates: Array<[number, number, number]>, durationDays = 1): HolidayRule =>
(date) => {
const parts = utcParts(date);
return dates.some(([year, month, day]) => {
if (parts.year !== year) {
return false;
}
const start = Date.UTC(year, month, day);
const current = Date.UTC(parts.year, parts.month, parts.day);
return current >= start && current < start + durationDays * DAY_MS;
});
};
const inYearWindow =
(
windows: Array<{
year: number;
month: number;
day: number;
duration: number;
}>,
): HolidayRule =>
(date) => {
const parts = utcParts(date);
const window = windows.find((entry) => entry.year === parts.year);
if (!window) {
return false;
}
const start = Date.UTC(window.year, window.month, window.day);
const current = Date.UTC(parts.year, parts.month, parts.day);
return current >= start && current < start + window.duration * DAY_MS;
};
const isFourthThursdayOfNovember: HolidayRule = (date) => {
const parts = utcParts(date);
if (parts.month !== 10) {
return false;
} // November
const firstDay = new Date(Date.UTC(parts.year, 10, 1)).getUTCDay();
const offsetToThursday = (4 - firstDay + 7) % 7; // 4 = Thursday
const fourthThursday = 1 + offsetToThursday + 21; // 1st + offset + 3 weeks
return parts.day === fourthThursday;
};
const HOLIDAY_RULES = new Map<string, HolidayRule>([
[HOLIDAY_TAGLINES.newYear, onMonthDay(0, 1)],
[
HOLIDAY_TAGLINES.lunarNewYear,
onSpecificDates(
[
[2025, 0, 29],
[2026, 1, 17],
[2027, 1, 6],
],
1,
),
],
[
HOLIDAY_TAGLINES.eid,
onSpecificDates(
[
[2025, 2, 30],
[2025, 2, 31],
[2026, 2, 20],
[2027, 2, 10],
],
1,
),
],
[
HOLIDAY_TAGLINES.diwali,
onSpecificDates(
[
[2025, 9, 20],
[2026, 10, 8],
[2027, 9, 28],
],
1,
),
],
[
HOLIDAY_TAGLINES.easter,
onSpecificDates(
[
[2025, 3, 20],
[2026, 3, 5],
[2027, 2, 28],
],
1,
),
],
[
HOLIDAY_TAGLINES.hanukkah,
inYearWindow([
{ year: 2025, month: 11, day: 15, duration: 8 },
{ year: 2026, month: 11, day: 5, duration: 8 },
{ year: 2027, month: 11, day: 25, duration: 8 },
]),
],
[HOLIDAY_TAGLINES.halloween, onMonthDay(9, 31)],
[HOLIDAY_TAGLINES.thanksgiving, isFourthThursdayOfNovember],
[HOLIDAY_TAGLINES.valentines, onMonthDay(1, 14)],
[HOLIDAY_TAGLINES.christmas, onMonthDay(11, 25)],
]);
function isTaglineActive(tagline: string, date: Date): boolean {
const rule = HOLIDAY_RULES.get(tagline);
if (!rule) {
return true;
}
return rule(date);
}
export interface TaglineOptions {
env?: NodeJS.ProcessEnv;
random?: () => number;
now?: () => Date;
mode?: TaglineMode;
}
export function activeTaglines(options: TaglineOptions = {}): string[] {
if (TAGLINES.length === 0) {
return [DEFAULT_TAGLINE];
}
const today = options.now ? options.now() : new Date();
const filtered = TAGLINES.filter((tagline) => isTaglineActive(tagline, today));
return filtered.length > 0 ? filtered : TAGLINES;
}
export function pickTagline(options: TaglineOptions = {}): string {
if (options.mode === "off") {
return "";
}
if (options.mode === "default") {
return DEFAULT_TAGLINE;
}
const env = options.env ?? process.env;
const override = env?.OPENCLAW_TAGLINE_INDEX;
if (override !== undefined) {
const parsed = Number.parseInt(override, 10);
if (!Number.isNaN(parsed) && parsed >= 0) {
const pool = TAGLINES.length > 0 ? TAGLINES : [DEFAULT_TAGLINE];
return pool[parsed % pool.length];
}
}
const pool = activeTaglines(options);
const rand = options.random ?? Math.random;
const index = Math.floor(rand() * pool.length) % pool.length;
return pool[index];
}
export { TAGLINES, HOLIDAY_RULES, DEFAULT_TAGLINE };