mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-15 20:10:42 +00:00
287 lines
11 KiB
TypeScript
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 };
|