pty-mcp-server: pty-mcp-server

[ ai, apache, program ] [ Propose Tags ] [ Report a vulnerability ]

README.md


[Skip to Readme]

Downloads

Maintainer's Corner

Package maintainers

For package maintainers and hackage trustees

Candidates

  • No Candidates
Versions [RSS] 0.0.1.0, 0.0.1.1, 0.0.2.0, 0.0.3.0, 0.0.4.0, 0.0.5.0, 0.0.6.0, 0.0.7.0, 0.0.8.0, 0.0.9.0, 0.1.0.0, 0.1.1.0, 0.1.2.0, 0.1.3.0, 0.1.4.0
Change log CHANGELOG.md
Dependencies base (>=4.18 && <5), optparse-applicative, pms-application-service, pms-domain-model, pms-domain-service, pms-infra-cmdrun, pms-infra-procspawn, pms-infra-serial, pms-infra-socket, pms-infra-watch, pms-infrastructure, pms-ui-notification, pms-ui-request, pms-ui-response, safe-exceptions [details]
License Apache-2.0
Copyright 2025-2026 phoityne.hs@gmail.com
Author phoityne.hs@gmail.com
Maintainer phoityne.hs@gmail.com
Category AI
Home page https://github.com/phoityne/pty-mcp-server
Bug tracker https://github.com/phoityne/pty-mcp-server/issues
Uploaded by phoityne_hs at 2025-09-23T07:07:25Z
Distributions
Executables pty-mcp-server
Downloads 110 total (31 in the last 30 days)
Rating (no votes yet) [estimated by Bayesian average]
Your Rating
  • λ
  • λ
  • λ
Status Docs not available [build log]
Last success reported on 2025-09-23 [all 1 reports]

Readme for pty-mcp-server-0.1.4.0

[back to package description]

pty-mcp-server


⚠️ Caution

Do not grant unrestricted control to AI.
Unsupervised use or misuse may lead to unintended consequences.
All AI systems must remain strictly under human oversight and control.
Use responsibly, with full awareness and at your own risk.


📘 Overview

pty-mcp-server is a Haskell implementation of an MCP (Model Context Protocol) server that enables AI agents to dynamically acquire and control PTY (pseudo-terminal) sessions, allowing interaction with real system environments through terminal-based interfaces.

The server communicates exclusively via standard input/output (stdio), ensuring simple and secure integration with MCP clients. Through this interface, AI agents can execute commands, retrieve system states, and apply configurations—just as a human operator would through a terminal.


🎯 Purpose

  • Provide AI agents with TTY-based control capabilities
  • Enable automated configuration, inspection, and operation using CLI tools
  • Facilitate AI-driven workflows for system development, diagnostics, and remote interaction
  • Allow AI agents to access and operate on systems beyond the reach of static scripts or APIs
  • Support Infrastructure as Code (IaC) scenarios requiring interactive or stateful terminal workflows
  • Assist in system integration across heterogeneous environments and legacy systems
  • Empower AI agents to support DevOps, IaC, and integration pipelines by operating tools that require human-like terminal interaction

🔧 Example Use Cases

  • Dynamic execution of CLI tools that require a PTY environment
    (e.g., embedded systems over serial or SSH-based terminals)
  • REPL automation: driving GHCi or other CLI-based interactive interpreters
  • Interactive debugging of Haskell applications or shell-based workflows
  • System diagnostics through scripted or interactive bash sessions
  • Remote server management using SSH
  • Hands-on system operation where CLI behavior cannot be emulated via non-interactive scripting
  • Network device interaction: configuring routers, switches, or appliances via console
  • AI-assisted IaC workflows: executing Terraform, Ansible, or shell-based deploy scripts that involve prompts, state reconciliation, or real-time input
  • AI-driven system integration testing across multiple environments and CLI tools
  • Legacy system automation where GUI/API is unavailable and only terminal interaction is supported

User Guide (Usage and Setup)

Features

pty-mcp-server provides the following built-in tools for powerful and flexible automation:

Available Tools

  • pty-connect
    Runs a command via a pseudo-terminal (pty) to interact with external tools or services, with optional arguments.

  • pty-terminate
    Forcefully terminates an active pseudo-terminal (PTY) connection.

  • pty-message
    pms-messages is a tool for sending structured instructions or commands to a running PTY session. It abstracts direct terminal input, allowing the LLM (MCP client) to interact with the PTY process in a controlled and programmable way.

  • pty-bash
    pty-bash is a tool that launches a bash shell in a pseudo terminal (PTY). It allows the LLM (MCP client) to interact with a real Linux shell in an interactive terminal (PTY). This enables AI to run system commands, collect information, and handle prompts or TUI-based tools as if operated by a human, making it effective for dynamic Linux-based automation and diagnostics.

  • pty-ssh
    Establishes an SSH session in a pseudo-terminal with the specified arguments, allowing interaction with remote systems.

  • pty-telnet
    Launches the telnet command within a pseudo-terminal (PTY) session. This allows interactive communication with a remote Telnet server, enabling the AI to respond to prompts such as 'login:' or 'Password:' just like a human user. The PTY environment ensures that the terminal behaves like a real TTY device, which is required for many Telnet servers.

  • pty-cabal
    Launches a cabal repl session within a specified project directory, loading a target Haskell file.
    Supports argument passing and live code interaction.

  • pty-stack
    Launches a stack repl session in a pseudo-terminal using the specified project directory, main source file, and arguments.

  • pty-ghci
    Launches a GHCi session in a pseudo-terminal using the specified project directory, main source file, and arguments.

  • proc-spawn
    Spawns an external process using the specified arguments and enables interactive communication via standard input and output. Unlike PTY-based execution, this communicates directly with the process using the runProcess function without allocating a pseudo-terminal. Suitable for non-TUI, stdin/stdout-based interactive programs.

  • proc-terminate
    Forcefully terminates a running process created via runProcess.

  • proc-message
    Sends structured text-based instructions or commands to a subprocess started with runProcess. It provides a programmable interface for interacting with the process via standard input.

  • proc-cmd
    The proc-cmd tool launches the Windows Command Prompt (cmd.exe) as a subprocess. It allows the AI to interact with the standard Windows shell environment, enabling execution of batch commands, file operations, and system configuration tasks in a familiar terminal interface.

  • proc-ps
    proc-ps launches the Windows PowerShell (powershell.exe) as a subprocess. It provides an interactive command-line environment where the AI can execute PowerShell commands, scripts, and system administration tasks. The shell is started with default options to keep it open and ready for further input.

  • proc-ssh
    proc-ssh launches an SSH client (ssh) as a subprocess using runProcess. It enables the AI to initiate remote connections to other systems via the Secure Shell protocol. The tool can be used to execute remote commands, access remote shells, or tunnel services over SSH. The required arguments field allows specifying the target user, host, and any SSH options (e.g., -p, -i, -L).

  • proc-telnet
    A tool that runs Telnet sessions by internally using PuTTY's plink executable. This enables interactive Telnet connections on Windows without requiring an external pseudo-terminal emulator like winpty. Users supply the Telnet command arguments, which are passed directly to plink to establish the session. (Note: plink.exe must be available in the system PATH.)

  • proc-plink
    A Windows tool that launches an interactive console application via plink, a command-line SSH and Telnet client. Suitable for executing SSH or Telnet sessions directly without needing an external PTY emulator. (Note: plink.exe must be available in the system PATH.)

  • socket-open
    This tool initiates a socket connection to the specified host and port.

  • socket-close
    This tool close active socket connection that was previously established using the 'socket-opne' tool.

  • socket-read
    Reads the specified number of bytes from the socket. The 'size' parameter indicates how many bytes to read.

  • socket-write
    Write a sequence of bytes to the socket

  • socket-message
    This tool sends a specified string to the active socket connection, then waits for a recognizable prompt from the remote side. Upon detecting the prompt, it captures and returns all output received prior to it.

  • socket-telnet
    A simple Telnet-like communication tool over raw TCP sockets. This tool connects to a specified host and port, sends and receives data, and removes any Telnet IAC (Interpret As Command) sequences from the communication stream. Note: This is a simplified Telnet implementation and does not support full Telnet protocol features.

  • serial-open
    Opens a serial port connection to a specified device with a given baud rate. Commonly used to access on-premises hardware or network devices via console.

  • serial-close
    This tool close active serial connection that was previously established using the 'serial-open' tool.

  • serial-read
    Reads the specified number of bytes from the serial. The 'size' parameter indicates how many bytes to read.

  • serial-write
    Write a sequence of bytes to the serial.

  • serial-message
    This tool sends a specified string to the active socket connection, then waits for a recognizable prompt from the remote side. Upon detecting the prompt, it captures and returns all output received prior to it.

  • Scriptable CLI Integration
    The pty-mcp-server supports execution of shell scripts associated with registered tools defined in tools-list.json. Each tool must be registered by name, and a corresponding shell script (.sh) should exist in the configured tools/ directory.

    This design supports AI-driven workflows by exposing tool interfaces through a predictable scripting mechanism. The AI can issue tool invocations by name, and the server transparently manages execution and interaction.
    To add a new tool:

    1. Create a shell script named your-tool.sh in the tools/ directory.
    2. Add an entry in tools-list.json with the name "your-tool" and appropriate metadata.
    3. No need to recompile or modify the server — tools are dynamically resolved by name.

    This separation of tool definitions (tools-list.json) and implementation (tools/your-tool.sh) ensures clean decoupling and simplifies extensibility.

Note:
Commands starting with pty- are not supported on Windows. These tools rely on POSIX-style pseudo terminals (PTY), which are not natively available in the Windows environment.

Running with Podman or Docker

You can build and run pty-mcp-server using either Podman or Docker.

Note: When running pty-mcp-server inside a Docker container, after establishing a pty connection, you will be operating within the container environment. This should be taken into account when interacting with the server.

1. Build the image

Clone the repository and navigate to the docker directory:

$ git clone https://github.com/phoityne/pty-mcp-server.git
$ cd pty-mcp-server/docker
$ podman build . -t pty-mcp-server-image
$

Ref : build.sh

2. Run the container

Run the server inside a container:

$ podman run --rm -i \
--name pty-mcp-server-container \
-v /path/to/dir:/path/to/dir \
--hostname pms-docker-container \
pty-mcp-server-image \
-y /path/to/dir/config.yaml
$

Ref : run.sh

Below is an example of how to configure mcp.json to run the MCP server within VSCode:

{
  "servers": {
    "pty-mcp-server": {
      "type": "stdio",
      "command": "/path/to/run.sh",
      "args": []
      /*
      "command": "podman",
      "args": [
        "run", "--rm", "-i",
        "--name", "pty-mcp-server-container",
        "-v", "/path/to/dir:/path/to/dir",
        "--hostname", "pms-docker-container",
        "pty-mcp-server-image",
        "-y", "/path/to/dir/config.yaml"
      ]
      */
    }
  }
}

Binary Installation

If you prefer to build it yourself, make sure the following requirements are met:

  • GHC >= 9.6

You can install pty-mcp-server using cabal:

$ cabal install pty-mcp-server

Installation via .dxt Package

You can also set up the tool using a pre-packaged .dxt file.
This method is suitable for quick installation into Claude Code or for manual setup via extraction.

🛠️ The .dxt package distribution is currently in preparation,
but you can check the latest status and download links at:
https://github.com/phoityne/pms-dxt

Binary Execution

The pty-mcp-server application is executed from the command line.

Usage

$ pty-mcp-server -y config.yaml

While the server can be launched directly from the command line, it is typically started and managed by development tools that integrate an MCP client—such as Visual Studio Code. These tools utilize the server to enable interactive and automated command execution via PTY sessions.

VSCode Integration: .vscode/mcp.json

To streamline development and server invocation from within Visual Studio Code, the project supports a .vscode/mcp.json configuration file.

This file defines how the pty-mcp-server should be launched in a development environment. Example configuration:

{
  "servers": {
    "pty-mcp-server": {
      "type": "stdio",
      "command": "pty-mcp-server",
      "args": ["-y", "/path/to/your/config.yaml"]
    }
  }
}

config.yaml Configuration (ref)

  • logDir:
    The directory path where log files will be saved. This includes standard output/error logs and logs from script executions.

  • logLevel:
    Sets the logging level. Examples include "Debug", "Info", and "Error".

  • toolsDir:
    Directory containing script files (shell scripts named after tool names, e.g., ping.sh). If a script matching the tool name exists here, it will be executed when the tool is called.
    This directory must also contain the tools-list.json file, which defines the available public tools and their metadata.

  • prompts:
    A list of prompt strings used to detect interactive command prompts. This allows the AI to identify when a command is awaiting input. Examples include "ghci>", "]$", "password:", etc.


Demonstrations

AI handles Binary Protocol Dialogues via pty-mcp-server

Demo socket telnet
Ref : socket-telnet-prompt

This video demonstrates a Telnet login sequence powered by the MCP prompt defined in socket-telnet-prompt.md. Using tools like socket-open, socket-read, socket-write, and socket-message, the AI performs Telnet negotiation, handles prompts, and submits credentials. Binary responses are parsed and displayed in human-readable form.

Network Device Version Check via Serial Connection — powered by pty-mcp-server.

Demo serial
Ref : serial-nw-setting-prompt

This video demonstrates how pty-mcp-server enables AI-assisted automation over a serial connection to a network device.

  1. Device Setup The user specifies the communication port and baud rate.
    Example: COM3, 9600 baud on Windows.

  2. Login Interaction The AI prompts for a username and password,
    and uses them to log in to the network device.

  3. Device Version Retrieval After logging in, the AI sends a command
    to retrieve the installed OS or firmware version.

  4. Online Version Check The AI accesses the official website to check the latest available version,
    and compares it with the installed version.

  5. Session Termination Once the check is complete, the AI logs out and cleanly closes the serial connection.

Demo: Watch AI Create and Launch a Web App from Scratch

Demo web service construct
Ref : Web Service Construction Agent Prompt

  1. [Scene 1: Overview & MCP Configuration]
    In this demo, we’ll show how an AI agent builds and runs a web service inside a Docker container using the pty-mcp-server.
    First, we configure mcp.json to launch the MCP server using a shell script.
    This script starts the Docker container where our PTY-based interaction will take place.

  2. [Scene 2: Docker Launch Configuration]
    The run.sh script includes volume mounts, hostname settings, and opens port 8080.
    This allows the container to expose a web service to the host system.

  3. [Scene 3: Starting the MCP Server]
    Now, the container is launched, and the pty-mcp-server is running inside it,
    ready to handle AI-driven requests through a pseudo-terminal.

  4. [Scene 4: Connecting the AI Agent]
    We open the chat interface and send a prompt designed for a web service builder agent.
    The AI connects to the container’s Bash session via PTY and begins its preparation.

  5. [Scene 5: Initial Setup Commands]
    Following the prompt, the AI starts by:

    • Creating a project folder
    • Moving into the working directory
  6. [Scene 6: AI Ready to Receive Instructions]
    Once the environment is ready, we instruct the AI to build a “Hello, world” web service.
    From here, the AI begins its autonomous construction process.

  7. [Scene 7: AI Executes Web Setup Commands]
    The AI proposes a series of terminal commands.
    As the user, we review and approve them one by one.
    Steps include:

    • Checking for Python
    • Installing Flask
    • Writing the source code (app.py) to serve “Hello, world”
    • Running the Flask server
    • Testing via curl http://localhost:8080 inside the container
  8. [Scene 8: Verifying from Outside the Container]
    To confirm external accessibility, we access the service from the host via port 8080. As expected, the response is: “Hello, world”

  9. [Scene 9: Reviewing the Execution History]
    Finally, we review the AI's actions step by step:

    • Initialized the Bash session and created the working directory
    • Set up the Python environment
    • Generated the Flask-based app.py
    • Launched the web server and validated its operation
  10. [Scene 10: Conclusion]
    This demonstrates how AI, combined with the PTY-MCP-Server and Docker,
    can automate real development tasks — interactively, intelligently, and reproducibly.

Demo: Docker Execution and Host SSH Access

Demo Docker

  1. MCP Configuration with Docke
    This is the mcp.json file. It defines the MCP server startup configuration. In this case, the pty-mcp-server will be launched using a shell script: run.sh. This script uses Podman to start the container.
  2. Starting the MCP Server
    Here is the run.sh script. It launches the Docker container using podman run, with the correct volume mount, hostname, and image tag. Once executed, the MCP server starts inside the container.
  3. Tool List
    Next, the list of tools exposed to the client is defined in tools-list.json. It includes three tools: pty-message, pty-ssh, a shell script named hostname.sh
  4. Tool Script Directory
    In config.yaml, the path to the script directory is defined. This is where tool scripts like hostname.sh should be placed
  5. Hostname Script
    The hostname.sh script simply runs the hostname command. It is executed as a tool within the container.
  6. Executing hostname from Chat
    Now, let’s run the hostname tool in the chat. This shows the name of the current host, which is the container.
    As expected, the output is: pms-docker-container This confirms that the command is executed inside the Docker container.
  7. Using pty-ssh to Access the Host
    Next, we use pty-ssh to establish a pty session with the host OS. SSH connection is attempted using host.docker.internal, which resolves to the Docker host.
    After confirming the host identity and entering the password, the login succeeds.
  8. Confirming Host Environment
    Now that we are connected to the host, we run: cat /etc/redhat-release
    This confirms that we are now in the host OS, which is CentOS 9.
    In contrast, the Docker container is running AlmaLinux 9.

Demo: Interactive Bash via PTY

Demo bash

  1. Configure bash-mcp-server in mcp.json
    In this file, register bash-mcp-server as an MCP server.
    Specify the command as pty-mcp-server and pass the configuration file config.yaml as an argument.
  2. Settings in config.yaml
    The config.yaml file defines the log directory, the directory for tools, and prompt detection patterns.
    These settings establish the environment for the AI to interact with bash through the PTY.
  3. Place tools-list.json in the toolsDir
    You need to place tools-list.json in the directory specified by toolsDir.
    This file declares the tools available to the AI, including pty-bash and pty-message.
  4. AI Connects to Bash and Selects Commands Autonomously
    The AI connects to bash through the pseudo terminal and decides which commands to execute based on the context.
  5. Confirming the Command Execution Results
    The output of the getenforce command shows whether SELinux is in Enforcing mode.
    This result appears on the terminal or in logs, allowing the user to verify the system status.

Demo: Shell Script Execution

Demo shellscript

  1. mcp.json Configuration
    Starts the pty-mcp-server in stdio mode, passing config.yaml as an argument.
  2. Overview of config.yaml
    Specifies log directory, tools directory, and prompt strings.
    The tools-list.json in toolsDir defines which tools are exposed.
  3. Role of tools-list.json
    Lists available script tools, with only the script_add tool registered here.
  4. Role and Naming Convention of the tools Folder
    Stores executable shell scripts called via the mcp server.
    The tool names in tools-list.json match the shell script filenames in this folder.
  5. Execution from VSCode GitHub Copilot
    Runs script_add.sh with the command #script_add 2 3, executing the addition.
  6. Confirming the Result
    Returns "5", indicating the operation was successful.

Demo: Haskell Debugging with cabal repl

Demo haskell cabal repl
Ref : haskell cabal debug prompt

  1. Target Code Overview
    A function in MyLib.hs is selected to inspect its runtime state using cabal repl and an AI-driven debug interface.
  2. MCP Server Initialization
    The MCP server is launched to allow structured interaction between the AI and the debugging commands.
  3. Debugger Prompt and Environment Setup
    The AI receives a prompt, starts cabal repl, and loads the module to prepare for runtime inspection.
  4. Debugging Execution Begins
    The target function is executed and paused at a predefined point for runtime observation.
  5. State Inspection and Output
    Runtime values and control flow are displayed to help verify logic and observe internal behavior.
  6. Summary
    Integration with pty-msp-server enables automated runtime inspection for Haskell applications.

Architecture Guide (Software Architecture and Technical Details)

Architectural Strategy

The architecture of the pty-mcp-server project is designed with medium-to-large scale systems in mind. Emphasis is placed on modularity, maintainability, and scalability, especially in environments involving multiple teams or organizations.

To achieve these goals, the system is structured as a collection of well-separated packages, each responsible for a specific concern or domain. This package-oriented design provides several strategic benefits.

The overall package structure adheres to the principles of Onion Architecture, reflecting a layered design that places the domain model at the core. Furthermore, the internal module structure within each package is also guided by a layered approach, maintaining clear separation between pure data definitions, domain services, and infrastructure concerns.

Role of pty-mcp-server as a Dependency Injector

In addition to managing REPL communication, pty-mcp-server is not merely an executable module, but also acts as a dependency injector for the entire system.

  • It is capable of referencing all relevant PMS packages, including those that it depends on.
  • This allows it to construct and wire together application components across multiple packages and modules in a unified manner.
  • By centralizing this dependency resolution, pty-mcp-server provides a single point of control over cross-cutting dependencies, improving visibility and control over the system architecture.

As a result, inter-package and inter-module dependencies can be centrally coordinated and managed, which promotes better encapsulation, reusability, and testability throughout the system.

Dependencies

This package depends on the following packages:

Rationale for Package Separation

  • Clear Interface Definition
    Each package exposes only its minimal, well-defined public API. This enforces clean module boundaries and reduces unintended dependencies between components.

  • Team and Vendor Ownership
    In larger projects, different teams or external vendors can own specific packages. Clear separation ensures well-defined responsibilities and supports collaborative development across organizational boundaries.

  • Repository and Release Independence
    Packages can be split into separate repositories and versioned independently. This allows for modular development and flexible release workflows, reducing build times and simplifying integration.

  • Improved Maintainability and Extensibility
    By isolating concerns, the impact of code changes is limited to relevant modules. This minimizes regressions and facilitates safe, incremental improvements over time.

Intended Use Cases

  • Medium-to-large scale enterprise systems involving multiple developers or teams
  • Modular systems with independent development and release cycles for components
  • Projects that require long-term maintainability, extensibility, and isolation of concerns

This architecture follows a layered and modular approach. Domain models, domain services, application logic, and infrastructure concerns are each encapsulated in their own package, enabling clean composition while preserving separation of responsibilities.

Deployment Diagram

Deployment Diagram

Package Structure

Package Structure

Thread Structure

Thread Structure