From harness-claude
Implements GOF Abstract Factory pattern in TypeScript for creating families of related objects via factory interfaces without concrete coupling. Use for swappable UI themes or product suites like DB+cache pairs.
npx claudepluginhub intense-visions/harness-engineering --plugin harness-claudeThis skill uses the workspace's default tool permissions.
> Create families of related objects through factory interfaces without coupling to concrete types.
Implements Abstract Factory pattern in JavaScript to create families of related objects without specifying concrete classes. Useful for runtime-swappable sets like UI themes with buttons and inputs.
Audits PHP Abstract Factory pattern implementations for family consistency, product hierarchy completeness, factory method coverage, and cross-family compatibility issues.
Mandates invoking relevant skills via tools before any response in coding sessions. Covers access, priorities, and adaptations for Claude Code, Copilot CLI, Gemini CLI.
Share bugs, ideas, or general feedback.
Create families of related objects through factory interfaces without coupling to concrete types.
Define the abstract factory and product interfaces first:
// Abstract product interfaces
interface Button {
render(): string;
onClick(handler: () => void): void;
}
interface TextField {
render(): string;
getValue(): string;
}
// Abstract factory interface
interface UIFactory {
createButton(label: string): Button;
createTextField(placeholder: string): TextField;
}
Implement concrete families:
// --- Dark theme family ---
class DarkButton implements Button {
constructor(private label: string) {}
render() {
return `<button class="dark">${this.label}</button>`;
}
onClick(handler: () => void) {
/* attach event */
}
}
class DarkTextField implements TextField {
private value = '';
constructor(private placeholder: string) {}
render() {
return `<input class="dark" placeholder="${this.placeholder}" />`;
}
getValue() {
return this.value;
}
}
class DarkThemeFactory implements UIFactory {
createButton(label: string): Button {
return new DarkButton(label);
}
createTextField(placeholder: string): TextField {
return new DarkTextField(placeholder);
}
}
// --- Light theme family ---
class LightButton implements Button {
constructor(private label: string) {}
render() {
return `<button class="light">${this.label}</button>`;
}
onClick(handler: () => void) {
/* attach event */
}
}
class LightTextField implements TextField {
private value = '';
constructor(private placeholder: string) {}
render() {
return `<input class="light" placeholder="${this.placeholder}" />`;
}
getValue() {
return this.value;
}
}
class LightThemeFactory implements UIFactory {
createButton(label: string): Button {
return new LightButton(label);
}
createTextField(placeholder: string): TextField {
return new LightTextField(placeholder);
}
}
Client code depends only on the abstract factory:
class LoginForm {
private submitButton: Button;
private emailField: TextField;
constructor(factory: UIFactory) {
this.submitButton = factory.createButton('Log In');
this.emailField = factory.createTextField('Email address');
}
render(): string {
return `<form>${this.emailField.render()}${this.submitButton.render()}</form>`;
}
}
// Swap families by changing the factory
const theme = process.env.THEME === 'dark' ? new DarkThemeFactory() : new LightThemeFactory();
const form = new LoginForm(theme);
console.log(form.render());
Cloud provider example (more realistic for backend):
interface StorageProvider {
upload(key: string, data: Buffer): Promise<string>;
download(key: string): Promise<Buffer>;
}
interface QueueProvider {
publish(topic: string, message: object): Promise<void>;
subscribe(topic: string, handler: (msg: object) => void): void;
}
interface CloudFactory {
createStorage(bucket: string): StorageProvider;
createQueue(): QueueProvider;
}
class AWSFactory implements CloudFactory {
createStorage(bucket: string): StorageProvider {
return new S3Storage(bucket);
}
createQueue(): QueueProvider {
return new SQSQueue();
}
}
class GCPFactory implements CloudFactory {
createStorage(bucket: string): StorageProvider {
return new GCSStorage(bucket);
}
createQueue(): QueueProvider {
return new PubSubQueue();
}
}
Abstract Factory vs. Factory Method: Factory Method uses inheritance — one factory method per subclass. Abstract Factory uses composition — inject a factory object that produces a whole family. Abstract Factory is the better choice when you have multiple products that vary together.
Adding a new product to an existing family breaks the Open/Closed Principle — every concrete factory must implement the new method. This is the main drawback. Mitigate by providing a default implementation in an abstract base factory.
Anti-patterns:
Testing tip: Inject a MockFactory that returns stub implementations for every product type. This decouples all component tests from real infrastructure.
refactoring.guru/design-patterns/abstract-factory