scenario-planner

Scenario Planner for Construction

Safety Notice

This listing is imported from skills.sh public index metadata. Review upstream SKILL.md and repository scripts before running.

Copy this and send it to your AI assistant to learn

Install skill "scenario-planner" with this command: npx skills add datadrivenconstruction/ddc_skills_for_ai_agents_in_construction/datadrivenconstruction-ddc-skills-for-ai-agents-in-construction-scenario-planner

Scenario Planner for Construction

Overview

Model different project scenarios to understand their impacts on cost, schedule, and resources. Compare alternatives, optimize decisions, and prepare for contingencies.

Business Case

Construction decisions require understanding trade-offs:

  • Design Alternatives: Which option is most cost-effective?

  • Schedule Compression: What's the cost of accelerating?

  • Resource Options: In-house vs. subcontractor?

  • Risk Scenarios: What if materials increase 20%?

Technical Implementation

from dataclasses import dataclass, field from typing import List, Dict, Any, Optional, Callable from datetime import datetime, timedelta import pandas as pd import numpy as np from copy import deepcopy

@dataclass class ScenarioParameter: name: str base_value: float unit: str min_value: Optional[float] = None max_value: Optional[float] = None description: str = ""

@dataclass class Scenario: id: str name: str description: str parameters: Dict[str, float] created_at: datetime = field(default_factory=datetime.now)

@dataclass class ScenarioResult: scenario_id: str scenario_name: str total_cost: float total_duration: int # days resource_requirements: Dict[str, float] risk_score: float key_metrics: Dict[str, float] warnings: List[str] comparison_to_base: Dict[str, float]

@dataclass class SensitivityResult: parameter: str values_tested: List[float] cost_impacts: List[float] duration_impacts: List[float] sensitivity_score: float

class ConstructionScenarioPlanner: """Scenario planning and what-if analysis for construction."""

def __init__(self, base_project: Dict):
    self.base_project = base_project
    self.parameters: Dict[str, ScenarioParameter] = {}
    self.scenarios: Dict[str, Scenario] = {}
    self.results: Dict[str, ScenarioResult] = {}
    self.cost_model: Optional[Callable] = None
    self.duration_model: Optional[Callable] = None
    self._setup_default_parameters()

def _setup_default_parameters(self):
    """Setup common construction scenario parameters."""
    default_params = [
        ScenarioParameter("labor_rate", 75, "$/hr", 50, 150, "Average labor rate"),
        ScenarioParameter("material_escalation", 0, "%", -10, 30, "Material cost change"),
        ScenarioParameter("productivity_factor", 1.0, "x", 0.5, 1.5, "Labor productivity multiplier"),
        ScenarioParameter("overtime_percentage", 0, "%", 0, 50, "Overtime work percentage"),
        ScenarioParameter("crew_size", 10, "workers", 5, 50, "Average crew size"),
        ScenarioParameter("work_days_per_week", 5, "days", 5, 7, "Working days per week"),
        ScenarioParameter("contingency_percentage", 10, "%", 5, 25, "Cost contingency"),
        ScenarioParameter("weather_delay_days", 0, "days", 0, 60, "Expected weather delays"),
        ScenarioParameter("permit_delay_days", 0, "days", 0, 90, "Expected permit delays"),
        ScenarioParameter("subcontractor_markup", 15, "%", 10, 30, "Subcontractor markup"),
    ]

    for param in default_params:
        self.parameters[param.name] = param

def add_parameter(self, param: ScenarioParameter):
    """Add custom parameter."""
    self.parameters[param.name] = param

def set_cost_model(self, model: Callable):
    """Set custom cost calculation model."""
    self.cost_model = model

def set_duration_model(self, model: Callable):
    """Set custom duration calculation model."""
    self.duration_model = model

def create_scenario(self, name: str, description: str,
                   parameter_changes: Dict[str, float]) -> Scenario:
    """Create a new scenario with parameter modifications."""
    # Start with base values
    params = {p.name: p.base_value for p in self.parameters.values()}

    # Apply changes
    for param_name, value in parameter_changes.items():
        if param_name in params:
            params[param_name] = value
        else:
            raise ValueError(f"Unknown parameter: {param_name}")

    scenario = Scenario(
        id=f"SCN-{len(self.scenarios) + 1:03d}",
        name=name,
        description=description,
        parameters=params
    )

    self.scenarios[scenario.id] = scenario
    return scenario

def calculate_cost(self, params: Dict[str, float]) -> float:
    """Calculate total project cost based on parameters."""
    if self.cost_model:
        return self.cost_model(self.base_project, params)

    # Default cost model
    base_cost = self.base_project.get('base_cost', 1000000)

    # Labor adjustments
    labor_factor = params['labor_rate'] / 75  # Normalized to base rate
    productivity_impact = 1 / params['productivity_factor']
    overtime_premium = 1 + (params['overtime_percentage'] / 100 * 0.5)

    labor_cost = base_cost * 0.4 * labor_factor * productivity_impact * overtime_premium

    # Material adjustments
    material_cost = base_cost * 0.35 * (1 + params['material_escalation'] / 100)

    # Equipment and other
    equipment_cost = base_cost * 0.15

    # Subcontractor
    sub_cost = base_cost * 0.1 * (1 + params['subcontractor_markup'] / 100)

    subtotal = labor_cost + material_cost + equipment_cost + sub_cost

    # Contingency
    total = subtotal * (1 + params['contingency_percentage'] / 100)

    return total

def calculate_duration(self, params: Dict[str, float]) -> int:
    """Calculate project duration based on parameters."""
    if self.duration_model:
        return self.duration_model(self.base_project, params)

    # Default duration model
    base_duration = self.base_project.get('base_duration', 365)

    # Crew size impact
    crew_factor = 10 / params['crew_size']  # Inverse relationship

    # Productivity impact
    productivity_factor = 1 / params['productivity_factor']

    # Work days impact
    workday_factor = 5 / params['work_days_per_week']

    # Overtime compression
    overtime_compression = 1 - (params['overtime_percentage'] / 100 * 0.3)

    calculated_duration = base_duration * crew_factor * productivity_factor * workday_factor * overtime_compression

    # Add delays
    delays = params['weather_delay_days'] + params['permit_delay_days']

    return int(calculated_duration + delays)

def evaluate_scenario(self, scenario: Scenario) -> ScenarioResult:
    """Evaluate a scenario and calculate results."""
    params = scenario.parameters

    total_cost = self.calculate_cost(params)
    total_duration = self.calculate_duration(params)

    # Calculate resource requirements
    resources = {
        'labor_hours': total_duration * params['crew_size'] * 8 * (params['work_days_per_week'] / 5),
        'peak_workers': params['crew_size'] * (1 + params['overtime_percentage'] / 100 * 0.5),
        'overtime_hours': total_duration * params['crew_size'] * 8 * params['overtime_percentage'] / 100,
    }

    # Calculate risk score (0-100)
    risk_factors = [
        params['overtime_percentage'] / 50 * 20,  # High overtime = higher risk
        (1 - params['productivity_factor']) * 20 if params['productivity_factor'] < 1 else 0,
        params['material_escalation'] / 30 * 15 if params['material_escalation'] > 0 else 0,
        (25 - params['contingency_percentage']) / 20 * 15,  # Low contingency = higher risk
    ]
    risk_score = min(sum(risk_factors), 100)

    # Key metrics
    cost_per_day = total_cost / total_duration
    cost_per_sf = total_cost / self.base_project.get('gross_area', 50000)

    key_metrics = {
        'cost_per_day': cost_per_day,
        'cost_per_sf': cost_per_sf,
        'labor_productivity': resources['labor_hours'] / total_duration,
    }

    # Warnings
    warnings = []
    if params['overtime_percentage'] > 30:
        warnings.append("High overtime may cause burnout and quality issues")
    if params['contingency_percentage'] < 8:
        warnings.append("Low contingency increases risk of budget overrun")
    if params['productivity_factor'] < 0.8:
        warnings.append("Low productivity factor may not be sustainable")

    # Compare to base scenario
    base_params = {p.name: p.base_value for p in self.parameters.values()}
    base_cost = self.calculate_cost(base_params)
    base_duration = self.calculate_duration(base_params)

    comparison = {
        'cost_change_pct': ((total_cost - base_cost) / base_cost) * 100,
        'cost_change_abs': total_cost - base_cost,
        'duration_change_pct': ((total_duration - base_duration) / base_duration) * 100,
        'duration_change_days': total_duration - base_duration,
    }

    result = ScenarioResult(
        scenario_id=scenario.id,
        scenario_name=scenario.name,
        total_cost=total_cost,
        total_duration=total_duration,
        resource_requirements=resources,
        risk_score=risk_score,
        key_metrics=key_metrics,
        warnings=warnings,
        comparison_to_base=comparison
    )

    self.results[scenario.id] = result
    return result

def run_sensitivity_analysis(self, parameter: str,
                              values: List[float] = None,
                              steps: int = 10) -> SensitivityResult:
    """Run sensitivity analysis on a single parameter."""
    if parameter not in self.parameters:
        raise ValueError(f"Unknown parameter: {parameter}")

    param = self.parameters[parameter]

    if values is None:
        min_val = param.min_value or param.base_value * 0.5
        max_val = param.max_value or param.base_value * 1.5
        values = np.linspace(min_val, max_val, steps).tolist()

    base_params = {p.name: p.base_value for p in self.parameters.values()}
    base_cost = self.calculate_cost(base_params)
    base_duration = self.calculate_duration(base_params)

    cost_impacts = []
    duration_impacts = []

    for val in values:
        test_params = base_params.copy()
        test_params[parameter] = val

        cost = self.calculate_cost(test_params)
        duration = self.calculate_duration(test_params)

        cost_impacts.append(((cost - base_cost) / base_cost) * 100)
        duration_impacts.append(((duration - base_duration) / base_duration) * 100)

    # Calculate sensitivity score (range of impact)
    cost_range = max(cost_impacts) - min(cost_impacts)
    duration_range = max(duration_impacts) - min(duration_impacts)
    sensitivity_score = (cost_range + duration_range) / 2

    return SensitivityResult(
        parameter=parameter,
        values_tested=values,
        cost_impacts=cost_impacts,
        duration_impacts=duration_impacts,
        sensitivity_score=sensitivity_score
    )

def compare_scenarios(self, scenario_ids: List[str] = None) -> pd.DataFrame:
    """Compare multiple scenarios side by side."""
    if scenario_ids is None:
        scenario_ids = list(self.scenarios.keys())

    data = []
    for sid in scenario_ids:
        if sid not in self.results:
            scenario = self.scenarios[sid]
            self.evaluate_scenario(scenario)

        result = self.results[sid]
        data.append({
            'Scenario': result.scenario_name,
            'Total Cost': f"${result.total_cost:,.0f}",
            'Duration (days)': result.total_duration,
            'Cost Change': f"{result.comparison_to_base['cost_change_pct']:+.1f}%",
            'Duration Change': f"{result.comparison_to_base['duration_change_days']:+.0f} days",
            'Risk Score': f"{result.risk_score:.0f}/100",
            'Cost/SF': f"${result.key_metrics['cost_per_sf']:.2f}",
        })

    return pd.DataFrame(data)

def find_optimal_scenario(self, objective: str = 'cost',
                          constraints: Dict[str, tuple] = None) -> Scenario:
    """Find optimal scenario given objective and constraints."""
    valid_results = []

    for sid, result in self.results.items():
        # Check constraints
        if constraints:
            meets_constraints = True
            if 'max_cost' in constraints and result.total_cost > constraints['max_cost']:
                meets_constraints = False
            if 'max_duration' in constraints and result.total_duration > constraints['max_duration']:
                meets_constraints = False
            if 'max_risk' in constraints and result.risk_score > constraints['max_risk']:
                meets_constraints = False

            if not meets_constraints:
                continue

        valid_results.append((sid, result))

    if not valid_results:
        return None

    # Sort by objective
    if objective == 'cost':
        valid_results.sort(key=lambda x: x[1].total_cost)
    elif objective == 'duration':
        valid_results.sort(key=lambda x: x[1].total_duration)
    elif objective == 'risk':
        valid_results.sort(key=lambda x: x[1].risk_score)
    elif objective == 'balanced':
        # Normalize and combine metrics
        valid_results.sort(key=lambda x: (
            x[1].total_cost / 1000000 +
            x[1].total_duration / 365 +
            x[1].risk_score / 100
        ))

    return self.scenarios[valid_results[0][0]]

def generate_report(self) -> str:
    """Generate scenario comparison report."""
    lines = ["# Scenario Analysis Report", ""]
    lines.append(f"**Project:** {self.base_project.get('name', 'Project')}")
    lines.append(f"**Generated:** {datetime.now().strftime('%Y-%m-%d %H:%M')}")
    lines.append(f"**Scenarios Analyzed:** {len(self.scenarios)}")
    lines.append("")

    # Comparison table
    lines.append("## Scenario Comparison")
    comparison = self.compare_scenarios()
    lines.append(comparison.to_markdown(index=False))
    lines.append("")

    # Best scenarios
    lines.append("## Optimal Scenarios")

    best_cost = self.find_optimal_scenario('cost')
    if best_cost:
        lines.append(f"- **Lowest Cost:** {best_cost.name}")

    best_duration = self.find_optimal_scenario('duration')
    if best_duration:
        lines.append(f"- **Shortest Duration:** {best_duration.name}")

    best_balanced = self.find_optimal_scenario('balanced')
    if best_balanced:
        lines.append(f"- **Best Balanced:** {best_balanced.name}")

    lines.append("")

    # Detailed results
    lines.append("## Detailed Results")
    for sid, result in self.results.items():
        lines.append(f"\n### {result.scenario_name}")
        lines.append(f"- **Cost:** ${result.total_cost:,.0f} ({result.comparison_to_base['cost_change_pct']:+.1f}%)")
        lines.append(f"- **Duration:** {result.total_duration} days ({result.comparison_to_base['duration_change_days']:+.0f})")
        lines.append(f"- **Risk Score:** {result.risk_score:.0f}/100")

        if result.warnings:
            lines.append("- **Warnings:**")
            for w in result.warnings:
                lines.append(f"  - ⚠️ {w}")

    return "\n".join(lines)

Quick Start

Define base project

base_project = { 'name': 'Office Building', 'base_cost': 5000000, 'base_duration': 365, 'gross_area': 50000 }

Initialize planner

planner = ConstructionScenarioPlanner(base_project)

Create scenarios

baseline = planner.create_scenario( "Baseline", "Standard approach with default parameters", {} )

accelerated = planner.create_scenario( "Accelerated Schedule", "Faster completion with overtime and larger crew", { 'overtime_percentage': 25, 'crew_size': 15, 'work_days_per_week': 6 } )

cost_optimized = planner.create_scenario( "Cost Optimized", "Lower cost with reduced contingency and smaller crew", { 'contingency_percentage': 7, 'crew_size': 8, 'subcontractor_markup': 12 } )

Evaluate all scenarios

for scenario in planner.scenarios.values(): result = planner.evaluate_scenario(scenario) print(f"{result.scenario_name}: ${result.total_cost:,.0f}, {result.total_duration} days")

Compare scenarios

comparison = planner.compare_scenarios() print(comparison)

Run sensitivity analysis

sensitivity = planner.run_sensitivity_analysis('material_escalation') print(f"Material escalation sensitivity: {sensitivity.sensitivity_score:.1f}")

Generate report

report = planner.generate_report() print(report)

Dependencies

pip install pandas numpy

Source Transparency

This detail page is rendered from real SKILL.md content. Trust labels are metadata-based hints, not a safety guarantee.

Related Skills

Related by shared tags or category signals.

Automation

cad-to-data

No summary provided by upstream source.

Repository SourceNeeds Review
Automation

drawing-analyzer

No summary provided by upstream source.

Repository SourceNeeds Review
Automation

dwg-to-excel

No summary provided by upstream source.

Repository SourceNeeds Review
Automation

cost-estimation-resource

No summary provided by upstream source.

Repository SourceNeeds Review