How to integrate Toggl MCP with LlamaIndex

This guide walks you through connecting Toggl to LlamaIndex using the Composio tool router. By the end, you'll have a working Toggl agent that can start a new time entry for coding, list all clients in your workspace, get details of your current running timer through natural language commands. This guide will help you understand how to give your LlamaIndex agent real control over a Toggl account through Composio's Toggl MCP server. Before we dive in, let's take a quick look at the key ideas and tools involved.

Toggl logoToggl
Api Key

Toggl is a time tracking platform for managing work hours and productivity. It helps individuals and teams monitor tasks, analyze time usage, and optimize workflows.

56 Tools

Introduction

This guide walks you through connecting Toggl to LlamaIndex using the Composio tool router. By the end, you'll have a working Toggl agent that can start a new time entry for coding, list all clients in your workspace, get details of your current running timer through natural language commands.

This guide will help you understand how to give your LlamaIndex agent real control over a Toggl account through Composio's Toggl MCP server.

Before we dive in, let's take a quick look at the key ideas and tools involved.

Also integrate Toggl with

TL;DR

Here's what you'll learn:
  • Set your OpenAI and Composio API keys
  • Install LlamaIndex and Composio packages
  • Create a Composio Tool Router session for Toggl
  • Connect LlamaIndex to the Toggl MCP server
  • Build a Toggl-powered agent using LlamaIndex
  • Interact with Toggl through natural language

What is LlamaIndex?

LlamaIndex is a data framework for building LLM applications. It provides tools for connecting LLMs to external data sources and services through agents and tools.

Key features include:

  • ReAct Agent: Reasoning and acting pattern for tool-using agents
  • MCP Tools: Native support for Model Context Protocol
  • Context Management: Maintain conversation context across interactions
  • Async Support: Built for async/await patterns

What is the Toggl MCP server, and what's possible with it?

The Toggl MCP server is an implementation of the Model Context Protocol that connects your AI agent and assistants like Claude, Cursor, etc directly to your Toggl account. It provides structured and secure access to your time tracking data, so your agent can perform actions like logging time entries, managing clients and projects, handling tags, and retrieving detailed activity reports on your behalf.

  • Automated time entry management: Let your agent start, stop, and create new time entries with precise details, making it easy to track your work hours hands-free.
  • Client and project organization: Easily add new clients or projects, fetch client details, or remove outdated clients to keep your workspace up to date and well-structured.
  • Real-time activity tracking: Ask your agent to retrieve the currently running time entry or list recent activities, so you always know where your time is going.
  • Tag management and organization: Automatically create or delete tags to categorize your time entries, helping you analyze how your time is spent across different tasks.
  • Comprehensive workspace administration: Have your agent create organizations, set up workspaces, and ensure all your time tracking infrastructure is ready to go without manual setup.

What is the Composio tool router, and how does it fit here?

What is Composio SDK?

Composio's Composio SDK helps agents find the right tools for a task at runtime. You can plug in multiple toolkits (like Gmail, HubSpot, and GitHub), and the agent will identify the relevant app and action to complete multi-step workflows. This can reduce token usage and improve the reliability of tool calls. Read more here: Getting started with Composio SDK

The tool router generates a secure MCP URL that your agents can access to perform actions.

How the Composio SDK works

The Composio SDK follows a three-phase workflow:

  1. Discovery: Searches for tools matching your task and returns relevant toolkits with their details.
  2. Authentication: Checks for active connections. If missing, creates an auth config and returns a connection URL via Auth Link.
  3. Execution: Executes the action using the authenticated connection.

Step-by-step Guide

Step by step10 STEPS
1

Prerequisites

Before you begin, make sure you have:
  • Python 3.8/Node 16 or higher installed
  • A Composio account with the API key
  • An OpenAI API key
  • A Toggl account and project
  • Basic familiarity with async Python/Typescript
2

Getting API Keys for OpenAI, Composio, and Toggl

OpenAI API key (OPENAI_API_KEY)
  • Go to the OpenAI dashboard
  • Create an API key if you don't have one
  • Assign it to OPENAI_API_KEY in .env
Composio API key and user ID
  • Log into the Composio dashboard
  • Copy your API key from Settings
    • Use this as COMPOSIO_API_KEY
  • Pick a stable user identifier (email or ID)
    • Use this as COMPOSIO_USER_ID
3

Installing dependencies

npm install @composio/llamaindex @llamaindex/openai @llamaindex/tools @llamaindex/workflow dotenv

Create a new Typescript project and install the necessary dependencies:

  • @composio/llamaindex: Composio's LlamaIndex integration
  • @llamaindex/openai: OpenAI LLM integration
  • @llamaindex/tools: MCP client for LlamaIndex
  • @llamaindex/workflow: Workflow framework for LlamaIndex
  • dotenv: Environment variable management
4

Set environment variables

bash
OPENAI_API_KEY=your-openai-api-key
COMPOSIO_API_KEY=your-composio-api-key
COMPOSIO_USER_ID=your-user-id

Create a .env file in your project root:

These credentials will be used to:

  • Authenticate with OpenAI's GPT-5 model
  • Connect to Composio's Tool Router
  • Identify your Composio user session for Toggl access
5

Import modules

import "dotenv/config";
import readline from "node:readline/promises";
import { stdin as input, stdout as output } from "node:process";

import { Composio } from "@composio/core";

import { mcp } from "@llamaindex/tools";
import { agent as createAgent } from "@llamaindex/workflow";
import { openai } from "@llamaindex/openai";

dotenv.config();

Create a new file called toggl_llamaindex_agent.ts and import the required modules:

Key imports:

  • dotenv.config loads .env at runtime
  • readline gives us a simple CLI chat loop
  • Composio is the main Composio SDK client
  • mcp connects to an MCP endpoint
  • createAgent builds a LlamaIndex agent
  • openai configures the LLM backend
6

Load environment variables and initialize Composio

const OPENAI_API_KEY = process.env.OPENAI_API_KEY;
const COMPOSIO_API_KEY = process.env.COMPOSIO_API_KEY;
const COMPOSIO_USER_ID = process.env.COMPOSIO_USER_ID;

if (!OPENAI_API_KEY) throw new Error("OPENAI_API_KEY is not set");
if (!COMPOSIO_API_KEY) throw new Error("COMPOSIO_API_KEY is not set");
if (!COMPOSIO_USER_ID) throw new Error("COMPOSIO_USER_ID is not set");

What's happening:

This ensures missing credentials cause early, clear errors before the agent attempts to initialise.

7

Create a Tool Router session and build the agent function

async function buildAgent() {

  console.log(`Initializing Composio client...${COMPOSIO_USER_ID!}...`);
  console.log(`COMPOSIO_USER_ID: ${COMPOSIO_USER_ID!}...`);

  const composio = new Composio({
    apiKey: COMPOSIO_API_KEY,
    provider: new LlamaindexProvider(),
  });

  const session = await composio.create(
    COMPOSIO_USER_ID!,
    {
      toolkits: ["toggl"],
    },
  );

  const mcpUrl = session.mcp.url;
  console.log(`Composio Tool Router MCP URL: ${mcpUrl}`);

  const server = mcp({
    url: mcpUrl,
    clientName: "composio_tool_router_with_llamaindex",
    requestInit: {
      headers: {
        "x-api-key": COMPOSIO_API_KEY!,
      },
    },
    // verbose: true,
  });

  const tools = await server.tools();

  const llm = openai({ apiKey: OPENAI_API_KEY, model: "gpt-5" });

  const agent = createAgent({
    name: "composio_tool_router_with_llamaindex",
        description : "An agent that uses Composio Tool Router MCP tools to perform actions.",
    systemPrompt:
      "You are a helpful assistant connected to Composio Tool Router."+
"Use the available tools to answer user queries and perform Toggl actions." ,
    llm,
    tools,
  });

  return agent;
}

What's happening here:

  • We create a Composio client using your API key and configure it with the LlamaIndex provider
  • We then create a tool router MCP session for your user, specifying the toolkits we want to use (in this case, toggl)
  • The session returns an MCP HTTP endpoint URL that acts as a gateway to all your configured tools
  • LlamaIndex will connect to this endpoint to dynamically discover and use the available Toggl tools.
  • The MCP tools are mapped to LlamaIndex-compatible tools and plug them into the Agent.
8

Create an interactive chat loop

async function chatLoop(agent: ReturnType<typeof createAgent>) {
  const rl = readline.createInterface({ input, output });

  console.log("Type 'quit' or 'exit' to stop.");

  while (true) {
    let userInput: string;

    try {
      userInput = (await rl.question("\nYou: ")).trim();
    } catch {
      console.log("\nAgent: Bye!");
      break;
    }

    if (!userInput) {
      continue;
    }

    const lower = userInput.toLowerCase();
    if (lower === "quit" || lower === "exit") {
      console.log("Agent: Bye!");
      break;
    }

    try {
      process.stdout.write("Agent: ");

      const stream = agent.runStream(userInput);
      let finalResult: any = null;

      for await (const event of stream) {
        // The event.data contains the streamed content
        const data: any = event.data;

        // Check for streaming delta content
        if (data?.delta) {
          process.stdout.write(data.delta);
        }

        // Store final result for fallback
        if (data?.result || data?.message) {
          finalResult = data;
        }
      }

      // If no streaming happened, show the final result
      if (finalResult) {
        const answer =
          finalResult.result ??
          finalResult.message?.content ??
          finalResult.message ??
          "";
        if (answer && typeof answer === "string" && !answer.includes("[object")) {
          process.stdout.write(answer);
        }
      }

      console.log(); // New line after streaming completes
    } catch (err: any) {
      console.error("\nAgent error:", err?.message ?? err);
    }
  }

  rl.close();
}

What's happening:

  • We're creating a direct terminal interface to chat with Toggl
  • The LLM's responses are streamed to the CLI for faster interaction.
  • The agent uses context to maintain conversation history
  • The agent processes the request, selects appropriate Toggl tools, and returns a result
  • We extract the answer from the result data structure and display it to the user
  • You can type 'quit' or 'exit' to stop the chat loop gracefully
  • Agent responses and any errors are streamed in a clear, readable format
9

Define the main entry point

async function main() {
  try {
    const agent = await buildAgent();
    await chatLoop(agent);
  } catch (err) {
    console.error("Failed to start agent:", err);
    process.exit(1);
  }
}

main();

What's happening here:

  • We're orchestrating the entire application flow
  • The agent gets built with proper error handling
  • Then we kick off the interactive chat loop so you can start talking to Toggl
10

Run the agent

npx ts-node llamaindex-agent.ts

When prompted, authenticate and authorise your agent with Toggl, then start asking questions.

Complete Code

Here's the complete code to get you started with Toggl and LlamaIndex:

import "dotenv/config";
import readline from "node:readline/promises";
import { stdin as input, stdout as output } from "node:process";

import { Composio } from "@composio/core";
import { LlamaindexProvider } from "@composio/llamaindex";

import { mcp } from "@llamaindex/tools";
import { agent as createAgent } from "@llamaindex/workflow";
import { openai } from "@llamaindex/openai";

dotenv.config();

const OPENAI_API_KEY = process.env.OPENAI_API_KEY;
const COMPOSIO_API_KEY = process.env.COMPOSIO_API_KEY;
const COMPOSIO_USER_ID = process.env.COMPOSIO_USER_ID;

if (!OPENAI_API_KEY) {
    throw new Error("OPENAI_API_KEY is not set in the environment");
  }
if (!COMPOSIO_API_KEY) {
    throw new Error("COMPOSIO_API_KEY is not set in the environment");
  }
if (!COMPOSIO_USER_ID) {
    throw new Error("COMPOSIO_USER_ID is not set in the environment");
  }

async function buildAgent() {

  console.log(`Initializing Composio client...${COMPOSIO_USER_ID!}...`);
  console.log(`COMPOSIO_USER_ID: ${COMPOSIO_USER_ID!}...`);

  const composio = new Composio({
    apiKey: COMPOSIO_API_KEY,
    provider: new LlamaindexProvider(),
  });

  const session = await composio.create(
    COMPOSIO_USER_ID!,
    {
      toolkits: ["toggl"],
    },
  );

  const mcpUrl = session.mcp.url;
  console.log(`Composio Tool Router MCP URL: ${mcpUrl}`);

  const server = mcp({
    url: mcpUrl,
    clientName: "composio_tool_router_with_llamaindex",
    requestInit: {
      headers: {
        "x-api-key": COMPOSIO_API_KEY!,
      },
    },
    // verbose: true,
  });

  const tools = await server.tools();

  const llm = openai({ apiKey: OPENAI_API_KEY, model: "gpt-5" });

  const agent = createAgent({
    name: "composio_tool_router_with_llamaindex",
    description:
      "An agent that uses Composio Tool Router MCP tools to perform actions.",
    systemPrompt:
      "You are a helpful assistant connected to Composio Tool Router."+
"Use the available tools to answer user queries and perform Toggl actions." ,
    llm,
    tools,
  });

  return agent;
}

async function chatLoop(agent: ReturnType<typeof createAgent>) {
  const rl = readline.createInterface({ input, output });

  console.log("Type 'quit' or 'exit' to stop.");

  while (true) {
    let userInput: string;

    try {
      userInput = (await rl.question("\nYou: ")).trim();
    } catch {
      console.log("\nAgent: Bye!");
      break;
    }

    if (!userInput) {
      continue;
    }

    const lower = userInput.toLowerCase();
    if (lower === "quit" || lower === "exit") {
      console.log("Agent: Bye!");
      break;
    }

    try {
      process.stdout.write("Agent: ");

      const stream = agent.runStream(userInput);
      let finalResult: any = null;

      for await (const event of stream) {
        // The event.data contains the streamed content
        const data: any = event.data;

        // Check for streaming delta content
        if (data?.delta) {
          process.stdout.write(data.delta);
        }

        // Store final result for fallback
        if (data?.result || data?.message) {
          finalResult = data;
        }
      }

      // If no streaming happened, show the final result
      if (finalResult) {
        const answer =
          finalResult.result ??
          finalResult.message?.content ??
          finalResult.message ??
          "";
        if (answer && typeof answer === "string" && !answer.includes("[object")) {
          process.stdout.write(answer);
        }
      }

      console.log(); // New line after streaming completes
    } catch (err: any) {
      console.error("\nAgent error:", err?.message ?? err);
    }
  }

  rl.close();
}

async function main() {
  try {
    const agent = await buildAgent();
    await chatLoop(agent);
  } catch (err: any) {
    console.error("Failed to start agent:", err?.message ?? err);
    process.exit(1);
  }
}

main();

Conclusion

You've successfully connected Toggl to LlamaIndex through Composio's Tool Router MCP layer. Key takeaways:
  • Tool Router dynamically exposes Toggl tools through an MCP endpoint
  • LlamaIndex's ReActAgent handles reasoning and orchestration; Composio handles integrations
  • The agent becomes more capable without increasing prompt size
  • Async Python provides clean, efficient execution of agent workflows
You can easily extend this to other toolkits like Gmail, Notion, Stripe, GitHub, and more by adding them to the toolkits parameter.
TOOLS

Supported Tools

Every Toggl action and event your agent gets out of the box.

Create Client

Tool to create a new client in a workspace.

Create Group

Tool to create a new group in a Toggl organization.

Create Invitation

Tool to send invitations to join a Toggl organization.

Create Organization

Tool to create a new organization with a default workspace in Toggl Track.

Create Project

Creates a new project in a Toggl workspace.

Create Tag

Tool to create a new tag in a workspace.

Create Time Entry

Tool to create a new time entry in the specified workspace.

Add User to Workspace Project

Tool to add a user to workspace project users.

Delete Toggl Client

Tool to delete a client in Toggl.

Delete Group

Tool to delete a group from a Toggl organization.

Delete Project Group

Tool to delete a project group from a Toggl workspace.

Delete Subscription

Tool to delete a webhook subscription in Toggl.

Delete Tag

Deletes a tag from a Toggl workspace.

Disable Weekly Report

Tool to disable weekly report email notifications.

Bulk Edit Time Entries

Tool to bulk edit multiple time entries in a workspace using JSON Patch operations.

Get All Plans

Tool to retrieve all available Toggl subscription plans and their features.

Get Client Details

Retrieves detailed information about a specific client in Toggl Track by its client ID and workspace ID.

Get Countries

Tool to retrieve all countries supported by Toggl.

Get Country Subdivisions

Tool to retrieve all subdivisions (states, provinces, regions) for a specific country in Toggl Track.

Get Currencies

Tool to retrieve the list of all currencies supported by Toggl Track.

Get Current Time Entry

Retrieves the currently running time entry for the authenticated user.

Get Event Filters

Retrieve the list of supported event filters for Toggl webhooks.

Get JWKS Keys

Retrieves the current JWKS (JSON Web Key Set) keyset used to sign JWT tokens.

List Clients

Retrieve a list of clients from a Toggl Track workspace with optional filtering by status and name.

Get My Location

Retrieves the authenticated user's last known location information including city, state, country, and coordinates.

Get My Quota

Tool to retrieve API rate limit quota for the authenticated user.

Get Organization Details

Retrieves detailed information about a specific Toggl organization including subscription plan, trial status, user count, and workspace settings.

Get Organization Groups

Retrieves all groups within a Toggl organization, including group members and workspace assignments.

Get Organization Users

Retrieves a list of users belonging to a Toggl organization.

Get Project Details

Tool to retrieve details of a specific project.

Get Projects

Tool to retrieve a list of projects from a Toggl workspace.

Get Public Subscription Plans

Tool to retrieve all publicly available subscription plans from Toggl.

Get Webhooks Status

Tool to retrieve the Toggl Webhooks server status.

Get Tags

Retrieve all tags in a Toggl workspace.

List Tasks

Tool to list tasks in a workspace or within a specific project.

Get Time Entries

Retrieve time entries for the authenticated user with flexible filtering options.

Get Time Entry

Tool to retrieve a specific time entry by its ID.

Get Timezone Offsets

Tool to retrieve all available timezone offsets from Toggl.

Get Timezones

Tool to retrieve all available timezones supported by Toggl Track.

Get User Clients

Retrieves all clients accessible to the authenticated user across all their workspaces.

Get User Preferences

Retrieves the authenticated user's preferences including timezone, date/time formats, notification settings, and enabled alpha/experimental features.

Get User Projects

Tool to retrieve all projects for the authenticated user.

Get User Tags

Tool to retrieve tags associated with the current user.

Get User Tasks

Retrieve all tasks across all workspaces accessible to the authenticated user.

Get User Workspaces

Tool to retrieve all workspaces the authenticated user belongs to.

Get Workspace Details

Retrieves comprehensive details and settings for a specific Toggl workspace by ID.

Get Workspace Logo

Tool to get workspace logo.

Get Workspace Preferences

Retrieves workspace preferences including the initial pricing plan and whether start/end times are hidden.

Get Workspace Users

Retrieves all users who belong to a specific Toggl workspace.

Stop Time Entry

Tool to stop a running time entry in a workspace.

Disable Product Emails

Tool to disable product emails for the authenticated user using a disable code.

Update Tag

Tool to update an existing tag in a specified workspace.

Send Demo Email

Tool to send a demo request email through Toggl's system.

Send Email to Contact

Tool to send an email to a contact via Toggl's smail service.

Send Smail Meet

Tool to send an email for meet.

Update Client

Updates an existing client in a Toggl workspace.

FAQ

Frequently asked questions

With a standalone Toggl MCP server, the agents and LLMs can only access a fixed set of Toggl tools tied to that server. However, with the Composio Tool Router, agents can dynamically load tools from Toggl and many other apps based on the task at hand, all through a single MCP endpoint.

Yes, you can. LlamaIndex fully supports MCP integration. You get structured tool calling, message history handling, and model orchestration while Tool Router takes care of discovering and serving the right Toggl tools.

Yes, absolutely. You can configure which Toggl scopes and actions are allowed when connecting your account to Composio. You can also bring your own OAuth credentials or API configuration so you keep full control over what the agent can do.

All sensitive data such as tokens, keys, and configuration is fully encrypted at rest and in transit. Composio is SOC 2 Type 2 compliant and follows strict security practices so your Toggl data and credentials are handled as safely as possible.

Start with Toggl.It takes 30 seconds.

Managed auth, hosted MCP servers, and every Toggl tool your agent needs.Free to start.

Start building