cwicr-escalation

CWICR Escalation Calculator

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 "cwicr-escalation" with this command: npx skills add datadrivenconstruction/ddc_skills_for_ai_agents_in_construction/datadrivenconstruction-ddc-skills-for-ai-agents-in-construction-cwicr-escalation

CWICR Escalation Calculator

Business Case

Problem Statement

Construction costs change over time:

  • Inflation affects all costs

  • Material prices fluctuate

  • Labor rates increase annually

  • Long projects need escalation

Solution

Time-based cost escalation using historical indices, projected rates, and category-specific escalation factors.

Business Value

  • Future pricing - Estimate costs at construction time

  • Budget planning - Account for inflation

  • Contract pricing - Escalation clauses

  • Historical analysis - Adjust past costs to current

Technical Implementation

import pandas as pd import numpy as np from typing import Dict, Any, List, Optional, Tuple from dataclasses import dataclass from datetime import datetime, date from dateutil.relativedelta import relativedelta from enum import Enum

class EscalationType(Enum): """Types of escalation.""" LABOR = "labor" MATERIAL = "material" EQUIPMENT = "equipment" GENERAL = "general"

@dataclass class EscalationIndex: """Escalation index for a period.""" period: str # YYYY-MM labor_index: float material_index: float equipment_index: float general_index: float

@dataclass class EscalationResult: """Result of escalation calculation.""" base_cost: float base_date: date target_date: date months: int escalation_rate: float escalation_amount: float escalated_cost: float by_category: Dict[str, Dict[str, float]]

Historical escalation rates (annual %)

HISTORICAL_RATES = { 2020: {'labor': 2.5, 'material': 1.8, 'equipment': 1.5, 'general': 2.0}, 2021: {'labor': 3.2, 'material': 8.5, 'equipment': 2.0, 'general': 4.5}, 2022: {'labor': 4.5, 'material': 12.0, 'equipment': 3.5, 'general': 7.0}, 2023: {'labor': 4.0, 'material': 5.0, 'equipment': 3.0, 'general': 4.0}, 2024: {'labor': 3.5, 'material': 3.0, 'equipment': 2.5, 'general': 3.0}, 2025: {'labor': 3.0, 'material': 2.5, 'equipment': 2.0, 'general': 2.5}, }

Material-specific escalation factors

MATERIAL_ESCALATION = { 'steel': 1.20, # Higher volatility 'lumber': 1.30, # High volatility 'concrete': 0.90, # Lower volatility 'copper': 1.25, # Commodity driven 'aluminum': 1.15, 'plastic': 1.10, 'glass': 0.95, 'default': 1.00 }

class CWICREscalation: """Calculate cost escalation over time."""

def __init__(self,
             cwicr_data: pd.DataFrame = None,
             custom_rates: Dict[int, Dict[str, float]] = None):
    self.cost_data = cwicr_data
    self.rates = custom_rates or HISTORICAL_RATES
    if cwicr_data is not None:
        self._index_data()

def _index_data(self):
    """Index cost data."""
    if 'work_item_code' in self.cost_data.columns:
        self._code_index = self.cost_data.set_index('work_item_code')
    else:
        self._code_index = None

def get_rate(self,
              year: int,
              category: EscalationType = EscalationType.GENERAL) -> float:
    """Get escalation rate for year and category."""
    year_rates = self.rates.get(year, self.rates.get(max(self.rates.keys())))
    return year_rates.get(category.value, year_rates.get('general', 3.0))

def calculate_compound_factor(self,
                               base_date: date,
                               target_date: date,
                               category: EscalationType = EscalationType.GENERAL) -> float:
    """Calculate compound escalation factor between dates."""

    if target_date <= base_date:
        return 1.0

    factor = 1.0
    current = base_date

    while current < target_date:
        year = current.year
        annual_rate = self.get_rate(year, category) / 100

        # Calculate months in this year
        year_end = date(year + 1, 1, 1)
        if target_date < year_end:
            months = (target_date.year - current.year) * 12 + target_date.month - current.month
        else:
            months = (year_end.year - current.year) * 12 + year_end.month - current.month

        # Apply monthly compound rate
        monthly_rate = (1 + annual_rate) ** (1/12) - 1
        factor *= (1 + monthly_rate) ** months

        current = year_end

    return factor

def escalate_cost(self,
                   base_cost: float,
                   base_date: date,
                   target_date: date,
                   cost_breakdown: Dict[str, float] = None) -> EscalationResult:
    """Escalate cost from base date to target date."""

    if cost_breakdown is None:
        cost_breakdown = {
            'labor': base_cost * 0.40,
            'material': base_cost * 0.45,
            'equipment': base_cost * 0.15
        }

    months = (target_date.year - base_date.year) * 12 + target_date.month - base_date.month

    # Escalate each category
    by_category = {}
    total_escalated = 0

    for category, amount in cost_breakdown.items():
        esc_type = EscalationType.LABOR if category == 'labor' else \
                   EscalationType.MATERIAL if category == 'material' else \
                   EscalationType.EQUIPMENT if category == 'equipment' else \
                   EscalationType.GENERAL

        factor = self.calculate_compound_factor(base_date, target_date, esc_type)
        escalated = amount * factor
        escalation = escalated - amount

        by_category[category] = {
            'base': round(amount, 2),
            'factor': round(factor, 4),
            'escalated': round(escalated, 2),
            'escalation': round(escalation, 2)
        }

        total_escalated += escalated

    total_escalation = total_escalated - base_cost
    esc_rate = (total_escalation / base_cost * 100) if base_cost > 0 else 0

    return EscalationResult(
        base_cost=round(base_cost, 2),
        base_date=base_date,
        target_date=target_date,
        months=months,
        escalation_rate=round(esc_rate, 2),
        escalation_amount=round(total_escalation, 2),
        escalated_cost=round(total_escalated, 2),
        by_category=by_category
    )

def escalate_estimate(self,
                       items: List[Dict[str, Any]],
                       base_date: date,
                       target_date: date) -> Dict[str, Any]:
    """Escalate entire estimate."""

    escalated_items = []
    total_base = 0
    total_escalated = 0

    for item in items:
        code = item.get('work_item_code', item.get('code'))
        qty = item.get('quantity', 0)

        # Get costs from CWICR
        labor = 0
        material = 0
        equipment = 0

        if self._code_index is not None and code in self._code_index.index:
            wi = self._code_index.loc[code]
            labor = float(wi.get('labor_cost', 0) or 0) * qty
            material = float(wi.get('material_cost', 0) or 0) * qty
            equipment = float(wi.get('equipment_cost', 0) or 0) * qty

        base = labor + material + equipment
        breakdown = {'labor': labor, 'material': material, 'equipment': equipment}

        result = self.escalate_cost(base, base_date, target_date, breakdown)

        escalated_items.append({
            'code': code,
            'base_cost': result.base_cost,
            'escalated_cost': result.escalated_cost,
            'escalation': result.escalation_amount
        })

        total_base += base
        total_escalated += result.escalated_cost

    return {
        'items': escalated_items,
        'total_base': round(total_base, 2),
        'total_escalated': round(total_escalated, 2),
        'total_escalation': round(total_escalated - total_base, 2),
        'escalation_rate': round((total_escalated - total_base) / total_base * 100, 2) if total_base > 0 else 0,
        'base_date': base_date,
        'target_date': target_date
    }

def project_future_costs(self,
                          base_cost: float,
                          base_date: date,
                          years_forward: int = 5,
                          annual_rate: float = None) -> pd.DataFrame:
    """Project costs for multiple future years."""

    projections = []
    current = base_cost

    for i in range(years_forward + 1):
        target = base_date + relativedelta(years=i)
        year = target.year

        if annual_rate is None:
            rate = self.get_rate(year)
        else:
            rate = annual_rate

        if i > 0:
            current = current * (1 + rate / 100)

        projections.append({
            'Year': year,
            'Date': target,
            'Annual Rate': f"{rate}%",
            'Projected Cost': round(current, 2),
            'Cumulative Escalation': round(current - base_cost, 2),
            'Cumulative %': round((current - base_cost) / base_cost * 100, 1)
        })

    return pd.DataFrame(projections)

def de_escalate_cost(self,
                      current_cost: float,
                      current_date: date,
                      base_date: date,
                      category: EscalationType = EscalationType.GENERAL) -> Dict[str, Any]:
    """De-escalate current cost back to base date."""

    factor = self.calculate_compound_factor(base_date, current_date, category)
    base_cost = current_cost / factor

    return {
        'current_cost': round(current_cost, 2),
        'current_date': current_date,
        'base_date': base_date,
        'de_escalation_factor': round(1 / factor, 4),
        'base_cost': round(base_cost, 2),
        'category': category.value
    }

def export_escalation(self,
                      result: EscalationResult,
                      output_path: str) -> str:
    """Export escalation to Excel."""

    with pd.ExcelWriter(output_path, engine='openpyxl') as writer:
        # Summary
        summary_df = pd.DataFrame([{
            'Base Cost': result.base_cost,
            'Base Date': result.base_date,
            'Target Date': result.target_date,
            'Months': result.months,
            'Escalation Rate': f"{result.escalation_rate}%",
            'Escalation Amount': result.escalation_amount,
            'Escalated Cost': result.escalated_cost
        }])
        summary_df.to_excel(writer, sheet_name='Summary', index=False)

        # By Category
        cat_df = pd.DataFrame([
            {
                'Category': cat,
                'Base': data['base'],
                'Factor': data['factor'],
                'Escalated': data['escalated'],
                'Escalation': data['escalation']
            }
            for cat, data in result.by_category.items()
        ])
        cat_df.to_excel(writer, sheet_name='By Category', index=False)

    return output_path

Quick Start

from datetime import date

Initialize escalation calculator

esc = CWICREscalation()

Escalate single cost

result = esc.escalate_cost( base_cost=1000000, base_date=date(2024, 1, 1), target_date=date(2026, 6, 1) )

print(f"Base Cost: ${result.base_cost:,.2f}") print(f"Escalated: ${result.escalated_cost:,.2f}") print(f"Escalation: {result.escalation_rate}%")

Common Use Cases

  1. Project Future Costs

projections = esc.project_future_costs( base_cost=5000000, base_date=date.today(), years_forward=5 ) print(projections)

  1. Escalate Estimate

cwicr = pd.read_parquet("ddc_cwicr_en.parquet") esc = CWICREscalation(cwicr)

items = [ {'work_item_code': 'CONC-001', 'quantity': 150}, {'work_item_code': 'STRL-002', 'quantity': 25} ]

escalated = esc.escalate_estimate( items, base_date=date(2024, 1, 1), target_date=date(2025, 12, 1) )

  1. De-escalate Historical Cost

base_cost = esc.de_escalate_cost( current_cost=1200000, current_date=date(2024, 6, 1), base_date=date(2020, 1, 1) ) print(f"2020 equivalent: ${base_cost['base_cost']:,.2f}")

Resources

  • GitHub: OpenConstructionEstimate-DDC-CWICR

  • DDC Book: Chapter 3.1 - Cost Escalation Methods

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