modular-saas-architecture/SKILL.md
Build SAAS platforms with pluggable business modules (Advanced Inventory, Restaurant, Pharmacy, etc.) that can be enabled/disabled per tenant without breaking the system. Use when designing modular SAAS features, implementing module toggles...
npx skillsauth add peterbamuhigire/skills-web-dev modular-saas-architectureInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
modular-saas-architecture or would be better handled by a more specific companion skill.references, examples, documentation only as needed.SKILL.md first, then load only the referenced deep-dive files that are necessary for the task.references/ directory for deep detail after reading the core workflow below.examples/ directory for concrete patterns when implementation shape matters.documentation/ directory for supporting implementation detail or migration notes.world-class-engineering for release gates and output standards.saas-erp-system-design when modules encode significant business workflows.database-design-engineering for schema ownership, tenancy, and migration safety.vibe-security-skill for security review.Architecture pattern for building SaaS platforms where business modules (Advanced Inventory, Restaurant, Pharmacy, Retail, etc.) can be independently enabled, disabled, or added without affecting other parts of the system.
Core Principles:
Security Baseline (Required): Always load and apply the Vibe Security Skill for any web app, API, or module implementation work.
📖 See references/implementation.md for full lifecycle code, testing patterns, and anti-patterns.
✅ Multi-tenant SaaS platforms with diverse customer needs ✅ Systems serving different industries (retail, healthcare, hospitality) ✅ Platforms with optional premium features ❌ Single-tenant applications or tightly coupled monolithic systems
┌─────────────────────────────────────────────────┐
│ CORE SYSTEM │
│ Authentication, Multi-tenant, Users, Billing, │
│ Audit Logs, Module Registry & Feature Flags │
└───────────────────┬─────────────────────────────┘
│
┌───────────────┼───────────────┐
▼ ▼ ▼
┌──────────┐ ┌──────────┐ ┌──────────┐
│ 🏪 Retail│ │🍽️ Restaurant│ │💊 Pharmacy│
│ POS Sales│ │Table Mgmt │ │Rx Mgmt │
│ Inventory│ │Orders │ │Drug DB │
│ Invoicing│ │Kitchen │ │Scripts │
└──────────┘ └──────────┘ └──────────┘
Per-Tenant Config:
Tenant A: Retail + Adv. Inventory (enabled)
Tenant B: Restaurant + Hospitality (enabled)
Tenant C: Pharmacy only (enabled)
modules/
└── advanced-inventory/
├── module.config.php # Module metadata, features, permissions, menu
├── permissions.php
├── routes.php
├── database/
│ └── schema.sql # Module tables (all franchise-scoped)
├── services/ # Business logic
├── models/
└── tests/
// modules/advanced-inventory/module.config.php
return [
'module_code' => 'ADV_INV',
'name' => 'Advanced Inventory',
'description' => 'Multi-location inventory with UOM conversions and transfers',
'version' => '1.0.0',
'requires' => [], // Empty = no dependencies
'features' => ['stock_items', 'uom_conversions', 'stock_transfers'],
'permissions' => ['VIEW_INVENTORY', 'MANAGE_STOCK', 'APPROVE_TRANSFERS'],
'tables' => ['tbl_stock_items', 'tbl_stock_item_uoms', 'tbl_stock_transfers'],
'menu' => [
['label' => 'Inventory', 'icon' => 'bi-boxes', 'items' => [
['label' => 'Stock Items', 'url' => '/stock-items-catalog.php'],
['label' => 'UOM Conversions', 'url' => '/advanced-inventory-uom.php'],
['label' => 'Stock Transfers', 'url' => '/stock-transfers.php'],
]]
],
'pricing' => ['type' => 'addon', 'price_monthly' => 29.99, 'trial_days' => 14],
];
class ModuleRegistry {
public function getEnabledModules(int $franchiseId): array {
$stmt = $this->db->prepare('
SELECT module_code, config FROM tbl_franchise_modules
WHERE franchise_id = ? AND is_enabled = 1
');
$stmt->execute([$franchiseId]);
return $stmt->fetchAll(PDO::FETCH_ASSOC);
}
public function isModuleEnabled(int $franchiseId, string $moduleCode): bool {
$stmt = $this->db->prepare('
SELECT is_enabled FROM tbl_franchise_modules
WHERE franchise_id = ? AND module_code = ?
');
$stmt->execute([$franchiseId, $moduleCode]);
return (bool) $stmt->fetchColumn();
}
}
// Blocking check — redirects if module disabled
function requireModuleAccess(string $moduleCode): void {
if (!isLoggedIn()) { header('Location: ./sign-in.php'); exit(); }
$stmt = $this->db->prepare('SELECT is_enabled FROM tbl_franchise_modules WHERE franchise_id = ? AND module_code = ?');
$stmt->execute([$_SESSION['franchise_id'], $moduleCode]);
if (!$stmt->fetchColumn()) {
header('Location: ./module-not-available.php?module=' . urlencode($moduleCode)); exit();
}
}
// Non-blocking check — for conditional UI
function hasModuleAccess(string $moduleCode): bool {
if (!isLoggedIn()) return false;
$stmt = $this->db->prepare('SELECT is_enabled FROM tbl_franchise_modules WHERE franchise_id = ? AND module_code = ?');
$stmt->execute([$_SESSION['franchise_id'], $moduleCode]);
return (bool) $stmt->fetchColumn();
}
// On every module page — at the top, before any output
requireModuleAccess('ADV_INV');
requirePermissionGlobal('VIEW_INVENTORY');
// Check before using an optional module
class OrderService {
public function createOrder(array $data) {
$order = $this->saveOrder($data);
// Use Advanced Inventory if enabled, else fallback
if (hasModuleAccess('ADV_INV')) {
(new AdvancedInventoryService())->updateInventory($order);
} else {
$this->updateBasicStock($order);
}
}
}
interface InventoryProvider {
public function checkStock(int $itemId, float $qty): bool;
public function decrementStock(int $itemId, float $qty): void;
}
class InventoryFactory {
public static function create(): InventoryProvider {
return hasModuleAccess('ADV_INV')
? new AdvancedInventory()
: new BasicInventory();
}
}
// Module A fires event — Module B optionally listens
class SalesModule {
public function completeSale(Sale $sale) {
$this->saveSale($sale);
EventDispatcher::dispatch('sale.completed', ['sale' => $sale]);
}
}
// modules/advanced-inventory/bootstrap.php
if (hasModuleAccess('ADV_INV')) {
EventDispatcher::listen('sale.completed', function($data) {
(new InventoryService())->decrementStock($data['sale']);
});
}
-- Module registry tables (core — always present)
CREATE TABLE tbl_modules (
module_code VARCHAR(50) PRIMARY KEY,
name VARCHAR(255) NOT NULL,
is_core BOOLEAN DEFAULT 0
);
CREATE TABLE tbl_franchise_modules (
id BIGINT AUTO_INCREMENT PRIMARY KEY,
franchise_id BIGINT NOT NULL,
module_code VARCHAR(50) NOT NULL,
is_enabled BOOLEAN DEFAULT 1,
enabled_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
disabled_at TIMESTAMP NULL,
config JSON,
UNIQUE KEY (franchise_id, module_code),
FOREIGN KEY (franchise_id) REFERENCES tbl_franchises(id),
FOREIGN KEY (module_code) REFERENCES tbl_modules(module_code)
);
-- Module tables: always include franchise_id for tenant isolation
CREATE TABLE tbl_stock_items (
id BIGINT AUTO_INCREMENT PRIMARY KEY,
franchise_id BIGINT NOT NULL,
name VARCHAR(255) NOT NULL,
FOREIGN KEY (franchise_id) REFERENCES tbl_franchises(id)
);
-- Cross-module FK: use NULL + application-level check (no hard FK to optional module)
CREATE TABLE tbl_sales (
id BIGINT AUTO_INCREMENT PRIMARY KEY,
franchise_id BIGINT NOT NULL,
restaurant_table_id BIGINT NULL -- Nullable! No FK to optional module table
);
// Menu constraints: max 5 submenus, max 6 items each, Bootstrap Icons on all entries
<nav>
<a href="/dashboard.php">Dashboard</a>
<?php if (hasModuleAccess('RETAIL')): ?>
<a href="/pos-sales.php">POS</a>
<?php endif; ?>
<?php if (hasModuleAccess('ADV_INV')): ?>
<div class="dropdown"><a href="#">Inventory</a>
<ul>
<li><a href="/stock-items-catalog.php">Stock Items</a></li>
<li><a href="/advanced-inventory-uom.php">UOM Conversions</a></li>
</ul>
</div>
<?php endif; ?>
</nav>
// Enable: validate dependencies → run migrations → insert row → create trial subscription → audit log
// Disable: check dependents → UPDATE is_enabled=0 → cancel subscription → audit log
// NEVER delete module data on disable — user may re-enable
UPDATE tbl_franchise_modules SET is_enabled = 0, disabled_at = NOW() WHERE franchise_id = ? AND module_code = ?;
DO:
franchise_id in module tableshasModuleAccess() for optional dependenciesDON'T:
requireModuleAccess() on every module pagehasModuleAccess() for optional dependency checkstbl_franchise_modules)hasModuleAccess()📖 See references/implementation.md for full lifecycle code, testing patterns, anti-patterns.
data-ai
Use when adding AI-powered analytics to a SaaS platform — semantic search over business data, natural language queries, trend detection, anomaly alerts, and AI-generated insights for dashboards. Covers embeddings, NL2SQL, and per-tenant analytics...
data-ai
Design AI-powered analytics dashboards — what metrics to show, how to display AI predictions and confidence, drill-down patterns, KPI cards, trend visualisation, AI Insights panels, export design, and role-based dashboard variants. Invoke when...
development
Use when designing, building, reviewing, or upgrading production software systems that must be secure, performant, maintainable, scalable, and user-centered. Apply before writing specs, code, architecture, APIs, databases, mobile apps, SaaS platforms, or ERP systems.
development
Professional web app UI using commercial templates (Tabler/Bootstrap 5) with strong frontend design direction when needed. Use for CRUD interfaces, dashboards, admin panels with SweetAlert2, DataTables, Flatpickr. Clone seeder-page.php, use...