toolbox-talk-generator

Toolbox Talk Generator

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

Toolbox Talk Generator

Overview

Automatically generate relevant safety toolbox talks based on daily work activities, weather conditions, recent incidents, and seasonal hazards. Support multiple languages for diverse crews.

"Daily toolbox talks reduce incidents by 30% when relevant to actual work" — DDC Community

How It Works

┌─────────────────────────────────────────────────────────────────┐ │ TOOLBOX TALK GENERATOR │ ├─────────────────────────────────────────────────────────────────┤ │ │ │ Inputs Generator Output │ │ ────── ───────── ────── │ │ 📅 Today's work → → 📋 Talk script │ │ 🌤️ Weather → 🤖 AI Engine → 📸 Visual aids │ │ ⚠️ Recent incidents → → ✅ Sign-in sheet │ │ 📆 Season/holiday → → 🌐 Translations │ │ │ └─────────────────────────────────────────────────────────────────┘

Technical Implementation

from dataclasses import dataclass, field from typing import List, Dict, Optional from enum import Enum from datetime import datetime, date import random

class HazardCategory(Enum): FALL_PROTECTION = "fall_protection" ELECTRICAL = "electrical" EXCAVATION = "excavation" SCAFFOLDING = "scaffolding" CRANE_RIGGING = "crane_rigging" CONFINED_SPACE = "confined_space" HOT_WORK = "hot_work" HAZMAT = "hazmat" HEAT_STRESS = "heat_stress" COLD_STRESS = "cold_stress" HOUSEKEEPING = "housekeeping" PPE = "ppe" HAND_TOOLS = "hand_tools" POWER_TOOLS = "power_tools" MATERIAL_HANDLING = "material_handling" TRAFFIC = "traffic" SILICA = "silica" NOISE = "noise"

@dataclass class ToolboxTalk: id: str title: str category: HazardCategory duration_minutes: int content: Dict[str, str] # sections: intro, hazards, controls, discussion, summary key_points: List[str] discussion_questions: List[str] date_generated: datetime = field(default_factory=datetime.now) language: str = "en"

@dataclass class TalkRecord: talk_id: str date: datetime project: str location: str presenter: str attendees: List[str] topics_covered: List[str] questions_raised: List[str] follow_up_actions: List[str]

class ToolboxTalkGenerator: """Generate contextual safety toolbox talks."""

# Talk templates by category
TALK_TEMPLATES = {
    HazardCategory.FALL_PROTECTION: {
        "title": "Fall Protection - Stay Safe at Heights",
        "intro": "Falls are the leading cause of death in construction. Today we'll review how to protect ourselves when working at heights.",
        "hazards": [
            "Unprotected edges and openings",
            "Improper ladder use",
            "Damaged or missing guardrails",
            "Incorrect harness use",
            "Unsecured tools and materials"
        ],
        "controls": [
            "Always use guardrails at 6 feet or above",
            "Inspect harness and lanyards before each use",
            "Maintain 3 points of contact on ladders",
            "Cover all floor openings",
            "Tether tools when working at height"
        ],
        "key_points": [
            "100% tie-off required above 6 feet",
            "Inspect fall protection daily",
            "Know your anchor points",
            "Report damaged equipment immediately"
        ],
        "discussion": [
            "Where are the fall hazards on our site today?",
            "What fall protection will you use?",
            "Have you inspected your equipment?"
        ]
    },
    HazardCategory.HEAT_STRESS: {
        "title": "Heat Stress Prevention",
        "intro": "Working in hot conditions can lead to serious illness. Today we'll discuss how to recognize and prevent heat-related illness.",
        "hazards": [
            "High temperatures and humidity",
            "Direct sun exposure",
            "Physical exertion",
            "Inadequate hydration",
            "Lack of acclimatization"
        ],
        "controls": [
            "Drink water every 15-20 minutes",
            "Take breaks in shade or cool areas",
            "Wear light, breathable clothing",
            "Know the signs of heat illness",
            "Use buddy system to watch each other"
        ],
        "key_points": [
            "Water, rest, shade - the three keys",
            "Don't wait until you're thirsty to drink",
            "Stop work if you feel dizzy or nauseous",
            "Acclimatize over 7-14 days"
        ],
        "discussion": [
            "Where are the water stations today?",
            "Where can you take a cool break?",
            "What are the symptoms of heat exhaustion?"
        ]
    },
    HazardCategory.ELECTRICAL: {
        "title": "Electrical Safety",
        "intro": "Electricity can kill instantly. Today we'll review how to work safely around electrical hazards.",
        "hazards": [
            "Overhead power lines",
            "Damaged cords and equipment",
            "Wet conditions",
            "Missing GFCIs",
            "Overloaded circuits"
        ],
        "controls": [
            "Maintain 10+ feet from power lines",
            "Inspect cords before use",
            "Use GFCIs for all power tools",
            "Never use damaged equipment",
            "Keep electrical away from water"
        ],
        "key_points": [
            "Assume all wires are energized",
            "Lock out/tag out before work",
            "Only qualified personnel do electrical work",
            "Report damaged equipment immediately"
        ],
        "discussion": [
            "Where are electrical hazards on site today?",
            "Are all your tools inspected?",
            "Where are the GFCIs located?"
        ]
    },
    HazardCategory.HOUSEKEEPING: {
        "title": "Good Housekeeping = Safe Workplace",
        "intro": "A clean site is a safe site. Poor housekeeping leads to trips, falls, and fires. Let's discuss keeping our work area organized.",
        "hazards": [
            "Debris and clutter in walkways",
            "Improper material storage",
            "Tangled cords and hoses",
            "Accumulated combustibles",
            "Blocked exits and access"
        ],
        "controls": [
            "Clean as you go throughout the day",
            "Store materials properly",
            "Route cords away from walkways",
            "Dispose of waste in proper containers",
            "Keep exits and aisles clear"
        ],
        "key_points": [
            "Clean up immediately after each task",
            "Everyone is responsible for housekeeping",
            "If you see it, fix it",
            "End each day with a clean work area"
        ],
        "discussion": [
            "What areas need attention today?",
            "Where should materials be stored?",
            "Who is responsible for end-of-day cleanup?"
        ]
    },
    HazardCategory.SCAFFOLDING: {
        "title": "Scaffold Safety",
        "intro": "Scaffolds provide safe access for work at height - but only when properly erected and used. Let's review scaffold safety.",
        "hazards": [
            "Incomplete or damaged scaffolds",
            "Missing guardrails or toeboards",
            "Overloading",
            "Improper access",
            "Unstable base"
        ],
        "controls": [
            "Only use tagged scaffolds (green tag)",
            "Check for complete guardrails",
            "Use proper access (ladder, stairs)",
            "Don't overload - check capacity",
            "Never modify scaffold yourself"
        ],
        "key_points": [
            "Green tag = safe to use",
            "Red/yellow tag = do not use",
            "Inspect before each use",
            "Report problems to supervisor"
        ],
        "discussion": [
            "Is the scaffold inspected and tagged?",
            "What is the load capacity?",
            "Where is the proper access point?"
        ]
    }
}

# Weather-related topic mapping
WEATHER_TOPICS = {
    "hot": [HazardCategory.HEAT_STRESS],
    "cold": [HazardCategory.COLD_STRESS],
    "rain": [HazardCategory.ELECTRICAL, HazardCategory.HOUSEKEEPING],
    "wind": [HazardCategory.CRANE_RIGGING, HazardCategory.SCAFFOLDING],
    "snow": [HazardCategory.COLD_STRESS, HazardCategory.HOUSEKEEPING]
}

# Activity-related topic mapping
ACTIVITY_TOPICS = {
    "concrete": [HazardCategory.SILICA, HazardCategory.MATERIAL_HANDLING],
    "steel": [HazardCategory.CRANE_RIGGING, HazardCategory.FALL_PROTECTION],
    "electrical": [HazardCategory.ELECTRICAL],
    "excavation": [HazardCategory.EXCAVATION],
    "roofing": [HazardCategory.FALL_PROTECTION, HazardCategory.HEAT_STRESS],
    "welding": [HazardCategory.HOT_WORK, HazardCategory.PPE],
    "demolition": [HazardCategory.SILICA, HazardCategory.HOUSEKEEPING],
    "painting": [HazardCategory.HAZMAT, HazardCategory.PPE]
}

def __init__(self):
    self.generated_talks: Dict[str, ToolboxTalk] = {}
    self.talk_records: List[TalkRecord] = []

def generate_talk(self, category: HazardCategory,
                 language: str = "en",
                 custom_points: List[str] = None) -> ToolboxTalk:
    """Generate toolbox talk for category."""
    template = self.TALK_TEMPLATES.get(category)

    if not template:
        # Generate generic talk
        template = self._generate_generic_template(category)

    talk_id = f"TBT-{datetime.now().strftime('%Y%m%d%H%M%S')}"

    content = {
        "intro": template["intro"],
        "hazards": "\n".join(f"• {h}" for h in template["hazards"]),
        "controls": "\n".join(f"• {c}" for c in template["controls"]),
        "summary": f"Remember: {template['key_points'][0]}"
    }

    key_points = template["key_points"].copy()
    if custom_points:
        key_points.extend(custom_points)

    talk = ToolboxTalk(
        id=talk_id,
        title=template["title"],
        category=category,
        duration_minutes=10,
        content=content,
        key_points=key_points,
        discussion_questions=template.get("discussion", []),
        language=language
    )

    self.generated_talks[talk_id] = talk
    return talk

def _generate_generic_template(self, category: HazardCategory) -> Dict:
    """Generate generic template for unmapped categories."""
    name = category.value.replace("_", " ").title()
    return {
        "title": f"{name} Safety",
        "intro": f"Today we'll discuss {name.lower()} safety and how to protect ourselves.",
        "hazards": [
            f"Common {name.lower()} hazards on site",
            "Lack of awareness",
            "Rushing or taking shortcuts",
            "Not following procedures"
        ],
        "controls": [
            "Follow all safety procedures",
            "Use required PPE",
            "Report hazards immediately",
            "Ask if unsure"
        ],
        "key_points": [
            "Safety first - always",
            "If unsure, ask your supervisor",
            "Report all hazards"
        ],
        "discussion": [
            f"What {name.lower()} hazards exist on site today?",
            "What controls will you use?",
            "Any questions or concerns?"
        ]
    }

def suggest_topics(self, weather: str = None,
                  activities: List[str] = None,
                  recent_incidents: List[str] = None) -> List[HazardCategory]:
    """Suggest relevant topics based on context."""
    suggestions = set()

    # Weather-based suggestions
    if weather:
        weather_lower = weather.lower()
        for condition, topics in self.WEATHER_TOPICS.items():
            if condition in weather_lower:
                suggestions.update(topics)

    # Activity-based suggestions
    if activities:
        for activity in activities:
            activity_lower = activity.lower()
            for key, topics in self.ACTIVITY_TOPICS.items():
                if key in activity_lower:
                    suggestions.update(topics)

    # Incident-based suggestions
    if recent_incidents:
        for incident in recent_incidents:
            incident_lower = incident.lower()
            if "fall" in incident_lower:
                suggestions.add(HazardCategory.FALL_PROTECTION)
            if "electric" in incident_lower:
                suggestions.add(HazardCategory.ELECTRICAL)
            if "struck" in incident_lower:
                suggestions.add(HazardCategory.MATERIAL_HANDLING)

    # Default suggestion if none
    if not suggestions:
        suggestions.add(HazardCategory.HOUSEKEEPING)

    return list(suggestions)

def generate_daily_talk(self, project: str,
                       weather: str,
                       activities: List[str],
                       recent_incidents: List[str] = None) -> ToolboxTalk:
    """Generate contextual daily toolbox talk."""
    topics = self.suggest_topics(weather, activities, recent_incidents)

    # Pick most relevant topic
    primary_topic = topics[0] if topics else HazardCategory.HOUSEKEEPING

    # Add context-specific points
    custom_points = []
    if weather:
        custom_points.append(f"Today's weather: {weather} - plan accordingly")
    if activities:
        custom_points.append(f"Today's focus: {', '.join(activities)}")

    talk = self.generate_talk(primary_topic, custom_points=custom_points)
    return talk

def format_talk_script(self, talk: ToolboxTalk) -> str:
    """Format talk as presenter script."""
    lines = [
        f"# {talk.title}",
        f"",
        f"**Duration:** {talk.duration_minutes} minutes",
        f"**Category:** {talk.category.value}",
        f"**Date:** {talk.date_generated.strftime('%Y-%m-%d')}",
        f"",
        f"---",
        f"",
        f"## Introduction",
        f"",
        talk.content["intro"],
        f"",
        f"## Hazards to Watch For",
        f"",
        talk.content["hazards"],
        f"",
        f"## How to Protect Yourself",
        f"",
        talk.content["controls"],
        f"",
        f"## Key Points to Remember",
        f"",
    ]

    for point in talk.key_points:
        lines.append(f"✓ {point}")

    lines.extend([
        f"",
        f"## Discussion Questions",
        f""
    ])

    for q in talk.discussion_questions:
        lines.append(f"❓ {q}")

    lines.extend([
        f"",
        f"---",
        f"",
        f"**Closing:** Work safe today. If you see something, say something. Any questions?"
    ])

    return "\n".join(lines)

def record_talk(self, talk_id: str, project: str, location: str,
               presenter: str, attendees: List[str],
               questions: List[str] = None,
               follow_ups: List[str] = None) -> TalkRecord:
    """Record completed toolbox talk."""
    if talk_id not in self.generated_talks:
        raise ValueError(f"Talk {talk_id} not found")

    talk = self.generated_talks[talk_id]

    record = TalkRecord(
        talk_id=talk_id,
        date=datetime.now(),
        project=project,
        location=location,
        presenter=presenter,
        attendees=attendees,
        topics_covered=[talk.category.value],
        questions_raised=questions or [],
        follow_up_actions=follow_ups or []
    )

    self.talk_records.append(record)
    return record

def get_attendance_summary(self, project: str = None,
                          start_date: datetime = None) -> Dict:
    """Get toolbox talk attendance summary."""
    records = self.talk_records

    if project:
        records = [r for r in records if r.project == project]
    if start_date:
        records = [r for r in records if r.date >= start_date]

    total_talks = len(records)
    total_attendees = sum(len(r.attendees) for r in records)
    unique_attendees = len(set(a for r in records for a in r.attendees))

    topics_covered = {}
    for r in records:
        for topic in r.topics_covered:
            topics_covered[topic] = topics_covered.get(topic, 0) + 1

    return {
        "total_talks": total_talks,
        "total_attendees": total_attendees,
        "unique_workers": unique_attendees,
        "avg_attendance": total_attendees / total_talks if total_talks else 0,
        "topics_covered": topics_covered
    }

Quick Start

Initialize generator

generator = ToolboxTalkGenerator()

Generate contextual daily talk

talk = generator.generate_daily_talk( project="Office Tower", weather="Hot, 95°F, sunny", activities=["concrete pour", "steel erection"], recent_incidents=["near miss - unsecured tool dropped"] )

Get formatted script

script = generator.format_talk_script(talk) print(script)

Record attendance

record = generator.record_talk( talk_id=talk.id, project="Office Tower", location="Level 5 staging area", presenter="John Smith", attendees=["Worker 1", "Worker 2", "Worker 3"], questions=["Where are the water coolers?"], follow_ups=["Add water station near grid C"] )

Get summary

summary = generator.get_attendance_summary() print(f"Total talks: {summary['total_talks']}") print(f"Unique workers trained: {summary['unique_workers']}")

Requirements

pip install (no external dependencies)

Integration with LLM

def generate_with_llm(topic: str, context: str) -> str: """Use LLM to generate custom toolbox talk.""" prompt = f""" Generate a 10-minute toolbox talk for construction workers about: {topic}

Context: {context}

Include:
1. Brief introduction (why this matters)
2. 4-5 specific hazards
3. 4-5 control measures
4. 3 key points to remember
5. 2-3 discussion questions

Keep language simple and direct. Workers have varying English levels.
"""

# Call your LLM API here
# response = llm.generate(prompt)
return response

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