Skip to content

Support Ticket Routing

This recipe shows how to take an incoming support ticket/call transcript then use an LLM to summarize the issue and route it to the correct person.

Mirascope Concepts Used

Background

Traditional machine learning techniques like text classification were previously used to solve routing. LLMs have enhanced routing by being able to better interpret nuances of inquiries as well as using client history and knowledge of the product to make more informed decisions.

Imitating a Company’s Database/Functionality

Fake Data

For both privacy and functionality purposes, these data types and functions in no way represent how a company’s API should actually look like. However, extrapolate on these gross oversimplifications to see how the LLM would interact with the company’s API.

User

Let’s create a User class to represent a customer as well as the function get_user_by_email() to imitate how one might search for the user in the database with some identifying information:

from pydantic import BaseModel

class User(BaseModel):
    name: str
    email: str
    past_purchases: list[str]
    past_charges: list[float]
    payment_method: str
    password: str
    security_question: str
    security_answer: str


def get_user_by_email(email: str):
    if email == "[email protected]":
        return User(
            name="John Doe",
            email="[email protected]",
            past_purchases=["TV", "Microwave", "Chair"],
            past_charges=[349.99, 349.99, 99.99, 44.99],
            payment_method="AMEX 1234 1234 1234 1234",
            password="password1!",
            security_question="Childhood Pet Name",
            security_answer="Piddles",
        )
    else:
        return None

Data Pulling Functions

Let’s also define some basic functions that one might expect a company to have for specific situations. get_sale_items() gets the items currently on sale, get_rewards() gets the rewards currently available to a user, get_billing_details() returns user data related to billing, and get_account_details() returns user data related to their account.

def get_sale_items():
    return "Sale items: we have a monitor at half off for $80!"

def get_rewards(user: User):
    if sum(user.past_charges) > 300:
        return "Rewards: for your loyalty, you get 10% off your next purchase!"
    else:
        return "Rewards: you have no rewards available right now."

def get_billing_details(user: User):
    return {
        "user_email": user.email,
        "user_name": user.name,
        "past_purchases": user.past_purchases,
        "past_charges": user.past_charges,
    }

def get_account_details(user: User):
    return {
        "user_email": user.email,
        "user_name": user.name,
        "password": user.password,
        "security_question": user.security_question,
        "security_answer": user.security_answer,
    }

Routing to Agent

Since we don’t have an actual endpoint to route to a live agent, let’s use this function route_to_agent() as a placeholder:

from typing import Literal

def route_to_agent(
    agent_type: Literal["billing", "sale", "support"], summary: str
) -> None:
    """Routes the call to an appropriate agent with a summary of the issue."""
    print(f"Routed to: {agent_type}\nSummary:\n{summary}")

Handling the Ticket

To handle the ticket, we will classify the issue of the ticket in one call, then use the classification to gather the corresponding context for a second call.

Classify the Transcript

Assume we have a basic transcript from the customer’s initial interactions with a support bot where they give some identifying information and their issue. We define a Pydantic BaseModel schema to classify the issue as well as grab the identifying information. calltype classifies the transcript into one of the three categories billing, sale, and support, and user_email will grab their email, assuming that’s what the bot asks for. The reasoning field will not be used, but forcing the LLM to give a reasoning for its classification choice aids in extraction accuracy, which can be shaky:

from pydantic import Field

class CallClassification(BaseModel):
    calltype: Literal["billing", "sale", "support"] = Field(
        ...,
        description="""The classification of the customer's issue into one of the 3: 
        'billing' for an inquiry about charges or payment methods,
        'sale' for making a purchase,
        'support' for general FAQ or account-related questions""",
    )
    reasoning: str = Field(
        ...,
        description="""A brief description of why the customer's issue fits into the\
              chosen category""",
    )
    user_email: str = Field(..., description="email of the user in the chat")

And we can extract information into this schema with the call classify_transcript():

from mirascope.core import openai, prompt_template

@openai.call(model="gpt-4o-mini", response_model=CallClassification)
@prompt_template(
    """
    Classify the following transcript between a customer and the service bot:
    {transcript}
    """
)
def classify_transcript(transcript: str): ...

Provide Ticket-Specific Context

Now, depending on the output of classify_transcript(), we would want to provide different context to the next call - namely, a billing ticket would necessitate the details from get_billing_details(), a sale ticket would want the output of get_sale_items() and get_rewards(), and a support_ticket would require get_account_details. We define a second call handle_ticket() which calls classify_transcript() and calls the correct functions for the scenario via dynamic configuration:

@openai.call(model="gpt-4o-mini", tools=[route_to_agent])
@prompt_template(
    """
    SYSTEM:
    You are an intermediary between a customer's interaction with a support chatbot and
    a real life support agent. Organize the context so that the agent can best
    facilitate the customer, but leave in details or raw data that the agent would need
    to verify a person's identity or purchase. Then, route to the appropriate agent.

    USER:
    {context}
    """
)
def handle_ticket(transcript: str) -> openai.OpenAIDynamicConfig:
    context = transcript
    call_classification = classify_transcript(transcript)
    user = get_user_by_email(call_classification.user_email)
    if isinstance(user, User):
        if call_classification.calltype == "billing":
            context += str(get_billing_details(user))
        elif call_classification.calltype == "sale":
            context += get_sale_items()
            context += get_rewards(user)
        elif call_classification.calltype == "support":
            context += str(get_account_details(user))
    else:
        context = "This person cannot be found in our system."

    return {"computed_fields": {"context": context}}

And there you have it! Let’s see how handle_ticket deals with each of the following transcripts:

billing_transcript = """
BOT: Please enter your email.
CUSTOMER: [email protected]
BOT: What brings you here today?
CUSTOMER: I purchased a TV a week ago but the charge is showing up twice on my bank \
statement. Can I get a refund?
"""

sale_transcript = """
BOT: Please enter your email.
CUSTOMER: [email protected]
BOT: What brings you here today?
CUSTOMER: I'm looking to buy a new monitor. Any discounts available?
"""

support_transcript = """
BOT: Please enter your email.
CUSTOMER: [email protected]
BOT: What brings you here today?
CUSTOMER: I forgot my site password and I'm also locked out of my email, how else can I
verify my identity?
"""

for transcript in [billing_transcript, sale_transcript, support_transcript]:
    response = handle_ticket(transcript)
    if tool := response.tool:
        tool.call()
# > Routed to: billing
#   Summary:
#   Customer, John Doe, is requesting a refund for a TV purchase showing a double charge on his bank statement, totaling $349.99.  
# > Routed to: sale
#   Summary:
#   Customer, John Doe, is interested in purchasing a new monitor and inquiring about discounts. Current promotion includes a monitor at half off for $80 and a 10% discount for loyalty.
# > Routed to: support
#   Summary:
#   Customer, John Doe, forgot their site password and is locked out of their email. Customer is requesting alternative methods for identity verification. They provided security question and answer: Childhood Pet Name - Piddles.

Additional Real-World Examples

  • IT Help Desk: Have LLM determine whether the user request is level 1, 2, or 3 support and route accordingly
  • Software-as-a-Service (SaaS) Companies: A question about how to use a specific feature might be routed to the product support team, while an issue with account access could be sent to the account management team.

When adapting this recipe to your specific use-case, consider the following:

- Update the `response_model` to more accurately reflect your use-case.
- Implement Pydantic `ValidationError` and Tenacity `retry` to improve reliability and accuracy.
- Evaluate the quality of extraction by using another LLM to verify classification accuracy.
- Use a local model like Ollama to protect company or other sensitive data.