email-construction

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

Email Generation for Construction

Overview

Generate professional construction emails with proper formatting, context, and attachments handling. Templates for common construction communication workflows.

Construction Use Cases

  1. RFI Response Email

Generate professional RFI response emails.

from dataclasses import dataclass from datetime import datetime from typing import Optional, List

@dataclass class RFIResponse: rfi_number: str project_name: str subject: str question: str response: str responder_name: str responder_title: str attachments: List[str] = None cc_list: List[str] = None

def generate_rfi_response_email(rfi: RFIResponse) -> dict: """Generate RFI response email."""

subject = f"RE: RFI #{rfi.rfi_number} - {rfi.subject}"

body = f"""Dear Project Team,

Please find below our response to RFI #{rfi.rfi_number}.

Project: {rfi.project_name} RFI Number: {rfi.rfi_number} Subject: {rfi.subject} Date: {datetime.now().strftime('%B %d, %Y')}


QUESTION: {rfi.question}


RESPONSE: {rfi.response}


Please proceed accordingly. If you have any questions regarding this response, please contact us.

{"Attachments:" + chr(10) + chr(10).join(f"- {a}" for a in rfi.attachments) if rfi.attachments else ""}

Best regards,

{rfi.responder_name} {rfi.responder_title} """

return {
    'subject': subject,
    'body': body,
    'cc': rfi.cc_list or [],
    'attachments': rfi.attachments or []
}

2. Submittal Transmittal Email

Generate submittal transmittal emails.

@dataclass class SubmittalTransmittal: submittal_number: str project_name: str spec_section: str description: str items: List[dict] action_required: str due_date: str sender_name: str sender_company: str

def generate_submittal_email(submittal: SubmittalTransmittal) -> dict: """Generate submittal transmittal email."""

subject = f"Submittal {submittal.submittal_number} - {submittal.spec_section} - {submittal.description}"

items_list = "\n".join(
    f"   {i+1}. {item['description']} ({item.get('copies', 1)} copies)"
    for i, item in enumerate(submittal.items)
)

body = f"""Dear Design Team,

Please find attached Submittal {submittal.submittal_number} for your review.

Project: {submittal.project_name} Submittal No: {submittal.submittal_number} Spec Section: {submittal.spec_section} Description: {submittal.description}

Items Transmitted: {items_list}

Action Required: {submittal.action_required} Response Requested By: {submittal.due_date}

Please review and return with your comments at your earliest convenience. If you have any questions, please don't hesitate to contact us.

Best regards,

{submittal.sender_name} {submittal.sender_company} """

return {
    'subject': subject,
    'body': body,
    'priority': 'normal'
}

3. Meeting Notice Email

Generate meeting invitation emails.

@dataclass class MeetingNotice: meeting_type: str # 'OAC', 'Subcontractor', 'Safety', 'Coordination' project_name: str date: str time: str location: str virtual_link: Optional[str] agenda_items: List[str] attendees: List[str] organizer_name: str

def generate_meeting_notice(meeting: MeetingNotice) -> dict: """Generate meeting notice email."""

subject = f"{meeting.meeting_type} Meeting - {meeting.project_name} - {meeting.date}"

agenda = "\n".join(f"   {i+1}. {item}" for i, item in enumerate(meeting.agenda_items))

location_info = meeting.location
if meeting.virtual_link:
    location_info += f"\n   Virtual Option: {meeting.virtual_link}"

body = f"""Dear Team,

You are invited to the {meeting.meeting_type} Meeting for {meeting.project_name}.

Meeting Details:

  • Date: {meeting.date}
  • Time: {meeting.time}
  • Location: {location_info}

Agenda: {agenda}

Attendees: {', '.join(meeting.attendees)}

Please confirm your attendance by replying to this email. If you cannot attend, please send a delegate and notify the organizer.

Regards,

{meeting.organizer_name} Project Manager """

return {
    'subject': subject,
    'body': body,
    'to': meeting.attendees,
    'calendar_invite': {
        'start': f"{meeting.date} {meeting.time}",
        'duration': 60,
        'location': meeting.location
    }
}

4. Change Order Notification

Generate change order notification emails.

@dataclass class ChangeOrderNotification: co_number: str project_name: str description: str amount: float schedule_impact: str reason: str status: str # 'Pending', 'Approved', 'Rejected' sender_name: str sender_title: str

def generate_change_order_email(co: ChangeOrderNotification) -> dict: """Generate change order notification email."""

subject = f"Change Order #{co.co_number} - {co.status} - {co.project_name}"

amount_str = f"${co.amount:,.2f}"
if co.amount < 0:
    amount_str = f"(${abs(co.amount):,.2f}) Credit"

body = f"""Dear Project Team,

This email is to notify you of Change Order #{co.co_number} for {co.project_name}.

Change Order Details:

  • CO Number: {co.co_number}
  • Status: {co.status}
  • Description: {co.description}

Financial Impact:

  • Amount: {amount_str}

Schedule Impact:

  • {co.schedule_impact}

Reason for Change: {co.reason}

{"Please review the attached documentation and provide your approval." if co.status == 'Pending' else ""} {"This change order has been approved. Please proceed accordingly." if co.status == 'Approved' else ""}

If you have any questions, please contact the project team.

Best regards,

{co.sender_name} {co.sender_title} """

return {
    'subject': subject,
    'body': body,
    'priority': 'high' if co.status == 'Pending' else 'normal',
    'flag': co.status == 'Pending'
}

5. Daily Report Email

Generate daily report distribution email.

@dataclass class DailyReportEmail: project_name: str report_date: str report_number: int weather: str workforce_total: int work_summary: List[str] issues: List[str] sender_name: str

def generate_daily_report_email(report: DailyReportEmail) -> dict: """Generate daily report distribution email."""

subject = f"Daily Report #{report.report_number} - {report.project_name} - {report.report_date}"

work_items = "\n".join(f"• {item}" for item in report.work_summary)
issues_text = "\n".join(f"• {issue}" for issue in report.issues) if report.issues else "None"

body = f"""Daily Construction Report

Project: {report.project_name} Date: {report.report_date} Report #: {report.report_number}


Weather: {report.weather} Total Workforce: {report.workforce_total} workers on-site


Work Completed: {work_items}


Issues/Delays: {issues_text}


Full report attached. Please contact the site office with any questions.

{report.sender_name} Site Superintendent """

return {
    'subject': subject,
    'body': body,
    'attachments': [f'Daily_Report_{report.report_number}.pdf']
}

6. Delay Notice Email

Generate formal delay notification.

@dataclass class DelayNotice: project_name: str contract_number: str delay_type: str # 'Excusable', 'Non-Excusable', 'Compensable' cause: str affected_activities: List[str] original_completion: str revised_completion: str days_impacted: int mitigation_plan: str sender_name: str sender_title: str

def generate_delay_notice_email(delay: DelayNotice) -> dict: """Generate formal delay notice email."""

subject = f"NOTICE OF DELAY - {delay.project_name} - {delay.days_impacted} Days"

activities = "\n".join(f"   - {a}" for a in delay.affected_activities)

body = f"""NOTICE OF DELAY

Project: {delay.project_name} Contract No: {delay.contract_number} Date: {datetime.now().strftime('%B %d, %Y')}


Dear Owner/Owner's Representative,

In accordance with the contract requirements, this letter serves as formal notice of a delay to the project schedule.

Delay Classification: {delay.delay_type}

Cause of Delay: {delay.cause}

Affected Activities: {activities}

Schedule Impact:

  • Original Completion Date: {delay.original_completion}
  • Revised Completion Date: {delay.revised_completion}
  • Calendar Days Impacted: {delay.days_impacted}

Mitigation Plan: {delay.mitigation_plan}

We request a meeting to discuss this matter and coordinate recovery efforts. Please contact us at your earliest convenience.

This notice is provided without prejudice to any rights or remedies available under the contract.

Respectfully,

{delay.sender_name} {delay.sender_title} """

return {
    'subject': subject,
    'body': body,
    'priority': 'high',
    'read_receipt': True,
    'delivery_receipt': True
}

Email Sending Integration

import smtplib from email.mime.text import MIMEText from email.mime.multipart import MIMEMultipart from email.mime.application import MIMEApplication

class ConstructionEmailSender: """Send construction emails via SMTP."""

def __init__(self, smtp_server: str, smtp_port: int, username: str, password: str):
    self.smtp_server = smtp_server
    self.smtp_port = smtp_port
    self.username = username
    self.password = password

def send(self, to: List[str], email_data: dict, from_addr: str = None):
    """Send email with optional attachments."""
    msg = MIMEMultipart()
    msg['From'] = from_addr or self.username
    msg['To'] = ', '.join(to)
    msg['Subject'] = email_data['subject']

    if email_data.get('cc'):
        msg['Cc'] = ', '.join(email_data['cc'])

    if email_data.get('priority') == 'high':
        msg['X-Priority'] = '1'

    msg.attach(MIMEText(email_data['body'], 'plain'))

    # Add attachments
    for attachment_path in email_data.get('attachments', []):
        with open(attachment_path, 'rb') as f:
            part = MIMEApplication(f.read())
            part.add_header('Content-Disposition', 'attachment',
                           filename=attachment_path.split('/')[-1])
            msg.attach(part)

    # Send
    with smtplib.SMTP(self.smtp_server, self.smtp_port) as server:
        server.starttls()
        server.login(self.username, self.password)

        recipients = to + email_data.get('cc', [])
        server.sendmail(msg['From'], recipients, msg.as_string())

Integration with DDC Pipeline

Example: Auto-generate RFI response email from RFI log

import pandas as pd

Load RFI log

rfi_log = pd.read_excel("RFI_Log.xlsx")

Get pending RFI

pending_rfi = rfi_log[rfi_log['status'] == 'Response Ready'].iloc[0]

Generate email

rfi = RFIResponse( rfi_number=pending_rfi['rfi_number'], project_name=pending_rfi['project'], subject=pending_rfi['subject'], question=pending_rfi['question'], response=pending_rfi['response'], responder_name='John Smith', responder_title='Project Architect', attachments=['SK-001.pdf'] )

email_data = generate_rfi_response_email(rfi) print(f"Subject: {email_data['subject']}") print(email_data['body'])

Email Templates Library

Common email templates for construction:

Template Use Case

RFI Response Responding to Requests for Information

Submittal Transmittal Sending submittals for review

Meeting Notice OAC, subcontractor, safety meetings

Change Order CO notifications and approvals

Daily Report Daily report distribution

Delay Notice Formal delay notifications

Payment Application Pay app submissions

Punch List Punch list item notifications

Closeout Warranty and closeout docs

Resources

  • Professional Email Writing: Keep emails concise and action-oriented

  • Construction Email Best Practices: Always include project name and reference numbers

  • Legal Considerations: Delay notices may have contractual requirements

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