Skip to main content

Pure Python 3270 emulator

Project description

Pure3270: Pure Python 3270 Terminal Emulation Library

Python Package Reports Linting GitHub Pages

Pure3270 is a self-contained, pure Python 3.8+ implementation of a 3270 terminal emulator, designed to emulate the functionality of the s3270 terminal emulator. It integrates seamlessly with the p3270 library through runtime monkey-patching, allowing you to replace p3270's dependency on the external s3270 binary without complex setup. The library uses standard asyncio for networking with no external telnet dependencies and supports TN3270 and TN3270E protocols, full 3270 emulation (screen buffer, fields, keyboard simulation), and optional SSL/TLS.

What's New in v0.2.1

This release marks a significant milestone with the completion of all high and medium priority features. Key enhancements include:

  • Complete s3270 Compatibility: Implementation of all missing s3270 actions including Compose(), Cookie(), Expect(), and Fail()
  • Full AID Support: Complete support for all PA (1-3) and PF (1-24) keys
  • Async Refactor: Complete async refactor with AsyncSession supporting connect, macro execution, and managed context
  • Protocol Enhancements: Complete TN3270E protocol support with printer session capabilities
  • Enhanced Field Handling: Improved field attribute handling and modification tracking for RMF/RMA commands
  • Comprehensive Macro Support: Advanced macro execution with conditional branching and variable substitution

For detailed release notes, see RELEASE_NOTES.md.

Key features:

  • Zero-configuration opt-in: Call enable_replacement() to patch p3270 automatically.
  • Standalone usage: Use Session or AsyncSession directly without p3270.
  • Pythonic API: Context managers, async support, and structured error handling.
  • Compatibility: Mirrors s3270 and p3270 interfaces with enhancements.

For architecture details, see architecture.md.

Installation

Pure3270 requires Python 3.8 or later. It is recommended to use a virtual environment for isolation.

1. Create and Activate Virtual Environment

Create a virtual environment in your project directory:

python -m venv .venv

Activate it:

  • On Unix/macOS:
    source .venv/bin/activate
    
  • On Windows:
    .venv\Scripts\activate
    

2. Install Pure3270

No external dependencies are required beyond the Python standard library for core usage.

For development (editable install):

pip install -e .

For distribution (from source):

pip install .

This uses the existing setup.py, which specifies no external dependencies. Deactivate the venv with deactivate when done.

Development Dependencies

For testing and linting, install additional tools:

pip install pytest-cov black flake8
  • pytest-cov: For coverage reporting (e.g., pytest --cov=pure3270).
  • black: For code formatting (e.g., black .).
  • flake8: For linting (e.g., flake8 .).

Exports

The main classes and functions are exported from the top-level module for easy import. From pure3270/__init__.py:

from pure3270 import Session, AsyncSession, enable_replacement

Quick Start Snippets

Enable Patching:

import pure3270
pure3270.enable_replacement()  # Patches p3270 for seamless integration

Synchronous Session:

from pure3270 import Session

with Session() as session:
    session.connect('your-host.example.com', port=23, ssl=False)
    session.send(b'key Enter')
    print(session.read())

Asynchronous Session:

import asyncio
from pure3270 import AsyncSession

async def main():
    async with AsyncSession() as session:
        await session.connect('your-host.example.com', port=23, ssl=False)
        await session.send(b'key Enter')
        print(await session.read())

asyncio.run(main())

Usage

Patching p3270 for Seamless Integration

To replace p3270's s3270 dependency with pure3270:

  1. Install p3270 in your venv: pip install p3270.
  2. Enable patching before importing p3270.

Example:

import pure3270
pure3270.enable_replacement()  # Applies global patches to p3270

import p3270
session = p3270.P3270Client()  # Now uses pure3270 under the hood
session.connect('your-host.example.com', port=23, ssl=False)
session.send(b'key Enter')
screen_text = session.read()
print(screen_text)
session.close()

This redirects p3270.P3270Client methods (__init__, connect, send, read) to pure3270 equivalents. Logs will indicate patching success.

Standalone Usage

Use pure3270 directly without p3270.

Synchronous Usage

From pure3270/session.py:

from pure3270 import Session

session = Session()
try:
    session.connect('your-host.example.com', port=23, ssl=False)
    session.send(b'key Enter')
    print(session.read())
finally:
    session.close()

Supports macros:

session.execute_macro('String(hello);key Enter')

Asynchronous Usage

From pure3270/session.py, AsyncSession provides async support for non-blocking operations.

Basic Connection and Send:

import asyncio
from pure3270 import AsyncSession

async def main():
    async with AsyncSession() as session:
        await session.connect('your-host.example.com', port=23, ssl=False)
        await session.send(b'key Enter')
        print(await session.read())

asyncio.run(main())

Executing Macros:

import asyncio
from pure3270 import AsyncSession

async def main():
    async with AsyncSession() as session:
        await session.connect('your-host.example.com', port=23, ssl=False)
        await session.execute_macro('String(hello);key Enter')
        print(await session.read())

asyncio.run(main())

Using Managed Context: The managed context manager ensures proper session lifecycle:

import asyncio
from pure3270 import AsyncSession

async def main():
    session = AsyncSession()
    async with session.managed():
        await session.connect('your-host.example.com', port=23, ssl=False)
        await session.send(b'key Enter')
        print(await session.read())
    # Session is automatically closed here

asyncio.run(main())

Handling Errors: Use try-except for robust error handling:

import asyncio
from pure3270 import AsyncSession, SessionError

async def main():
    try:
        async with AsyncSession() as session:
            await session.connect('your-host.example.com', port=23, ssl=False)
            await session.send(b'key Enter')
            print(await session.read())
    except SessionError as e:
        print(f"Session error: {e}")
    except Exception as e:
        print(f"Unexpected error: {e}")

asyncio.run(main())

See the examples/ directory for runnable scripts demonstrating these patterns.

API Reference

enable_replacement()

Top-level function to apply monkey patches to p3270 for transparent integration.

From pure3270/patching/patching.py:

def enable_replacement(
    patch_sessions: bool = True,
    patch_commands: bool = True,
    strict_version: bool = False
) -> MonkeyPatchManager:
    """
    Top-level API for zero-configuration opt-in patching.
    
    Applies global patches to p3270 for seamless pure3270 integration.
    Supports selective patching and fallback detection.
    
    :param patch_sessions: Patch session initialization and methods (default True).
    :param patch_commands: Patch command execution (default True).
    :param strict_version: Raise error on version mismatch (default False).
    :return: The MonkeyPatchManager instance for manual control.
    :raises Pure3270PatchError: If strict and patching fails.
    """

Returns a MonkeyPatchManager for advanced control (e.g., manager.unpatch()).

Session

Synchronous session handler for 3270 connections.

From pure3270/session.py:

class Session:
    """
    Synchronous 3270 session handler (wraps AsyncSession).
    """
    
    def __init__(self, host: Optional[str] = None, port: int = 23, ssl_context: Optional[Any] = None):
        """
        Initialize the Session.
        
        :param host: Hostname or IP.
        :param port: Port (default 23).
        :param ssl_context: SSL context for secure connections.
        """

    def connect(self, host: Optional[str] = None, port: Optional[int] = None, ssl_context: Optional[Any] = None) -> None:
        """
        Connect to the TN3270 host (sync).
        
        :param host: Hostname or IP.
        :param port: Port (default 23).
        :param ssl_context: SSL context for secure connections.
        :raises SessionError: If connection fails.
        """

    def send(self, data: bytes) -> None:
        """
        Send data to the host (sync).
        
        :param data: Data to send.
        :raises SessionError: If send fails.
        """

    def read(self, timeout: float = 5.0) -> bytes:
        """
        Read data from the host (sync).
        
        :param timeout: Read timeout in seconds.
        :return: Data received from host.
        :raises SessionError: If read fails.
        """

    def execute_macro(self, macro: str, vars: Optional[Dict[str, str]] = None) -> Dict[str, Any]:
        """
        Execute a macro (sync).
        
        :param macro: Macro to execute.
        :param vars: Variables for macro execution.
        :return: Macro execution results.
        """

    def close(self) -> None:
        """
        Close the session (sync).
        """

    @property
    def connected(self) -> bool:
        """
        Check if connected.
        """

Supports context manager: with Session() as session: ... (auto-closes on exit).

Additional properties:

  • tn3270_mode: bool - Check if TN3270 mode is active.
  • tn3270e_mode: bool - Check if TN3270E mode is active.
  • lu_name: Optional[str] - Get the bound LU name.

New s3270 Actions (High/Medium Priority Items)

Additional methods added for enhanced s3270 compatibility:

  • pf(self, n: int) -> None: Send PF (Program Function) key (1-24).
  • pa(self, n: int) -> None: Send PA (Program Attention) key (1-3).
  • compose(self, text: str) -> None: Compose special characters or key combinations.
  • cookie(self, cookie_string: str) -> None: Set HTTP cookie for web-based emulators.
  • expect(self, pattern: str, timeout: float = 10.0) -> bool: Wait for a pattern to appear on screen.
  • fail(self, message: str) -> None: Cause script to fail with a message.

AsyncSession

Asynchronous 3270 session handler.

From pure3270/session.py:

class AsyncSession:
    """Asynchronous 3270 session handler."""

    def __init__(
        self, host: Optional[str] = None, port: int = 23, ssl_context: Optional[Any] = None
    ):
        """
        Initialize the AsyncSession.

        :param host: Hostname or IP.
        :param port: Port (default 23).
        :param ssl_context: SSL context for secure connections.
        """

    async def connect(
        self, host: Optional[str] = None, port: Optional[int] = None, ssl_context: Optional[Any] = None
    ) -> None:
        """
        Connect to the TN3270 host.

        :param host: Hostname or IP.
        :param port: Port (default 23).
        :param ssl_context: SSL context for secure connections.
        :raises SessionError: If connection fails.
        """

    async def send(self, data: bytes) -> None:
        """
        Send data to the host.
        
        :param data: Data to send.
        :raises SessionError: If send fails.
        """

    async def read(self, timeout: float = 5.0) -> bytes:
        """
        Read data from the host.
        
        :param timeout: Read timeout in seconds.
        :return: Data received from host.
        :raises SessionError: If read fails.
        """

    async def execute_macro(self, macro: str, vars: Optional[Dict[str, str]] = None) -> Dict[str, Any]:
        """
        Execute a macro.
        
        :param macro: Macro to execute.
        :param vars: Variables for macro execution.
        :return: Macro execution results.
        """

    async def close(self) -> None:
        """Close the session."""

    @property
    def connected(self) -> bool:
        """Check if connected."""

    @asynccontextmanager
    async def managed(self):
        """Context manager for the session."""

Supports async context manager: async with session.managed(): ... (auto-closes on exit).

Additional properties:

  • tn3270_mode: bool - Check if TN3270 mode is active.
  • tn3270e_mode: bool - Check if TN3270E mode is active.
  • lu_name: Optional[str] - Get the bound LU name.

Other Exports

  • setup_logging(level: str = "INFO"): Configure logging for the library.
  • Exceptions: Pure3270Error, SessionError, ProtocolError, NegotiationError, ParseError, Pure3270PatchError.

For full details, refer to the source code or inline docstrings.

Testing

Pure3270 includes comprehensive tests in the tests/ directory, enhanced with edge cases for async operations, protocol handling, and patching.

Running Tests

Install dev dependencies (see Installation). Then:

pytest tests/

Coverage Reports

You can generate local coverage reports without using external services:

# Terminal report
pytest --cov=pure3270

# HTML report (creates interactive report in htmlcov/ directory)
pytest --cov=pure3270 --cov-report=html

# Detailed terminal report showing line numbers missing coverage
pytest --cov=pure3270 --cov-report=term-missing

# XML report (useful for CI/CD integration)
pytest --cov=pure3270 --cov-report=xml

You can also combine multiple report formats:

pytest --cov=pure3270 --cov-report=html --cov-report=term-missing

For linting:

black . --check
flake8 .

CI Setup

This project uses GitHub Actions for continuous integration. The workflows are defined in .github/workflows/:

  1. python-package.yml - Runs tests and linting across multiple Python versions
  2. reports.yml - Generates and publishes coverage and linting reports
  3. python-publish.yml - Publishes releases to PyPI

Reports are automatically generated on each push and pull request. Coverage reports are published to GitHub Pages for easy access.

name: CI

on: [push, pull_request]

jobs:
  test:
    runs-on: ubuntu-latest
    steps:
    - uses: actions/checkout@v2
    - name: Set up Python
      uses: actions/setup-python@v2
      with:
        python-version: 3.8
    - name: Install dependencies
      run: |
        python -m pip install --upgrade pip
        pip install -e .[dev]
    - name: Run tests with coverage
      run: pytest tests/ --cov=pure3270 --cov-report=xml --cov-report=html
    - name: Lint
      run: |
        black . --check
        flake8 .
    - name: Archive code coverage results
      uses: actions/upload-artifact@v4
      with:
        name: coverage-report
        path: htmlcov/

This runs tests, coverage, and linting on push/PR without requiring external coverage services. Coverage reports are uploaded as artifacts and published to GitHub Pages for the main branch.

Contribution Guidelines

Contributions are welcome! Please follow these steps:

  1. Fork the repository and create a feature branch.
  2. Install dev dependencies and run tests/linting locally.
  3. Make changes and add tests for new features.
  4. Ensure code passes black formatting and flake8 linting.
  5. Submit a pull request with a clear description of changes.

See the tests for examples. For major changes, open an issue first.

Migration Guide from s3270 / p3270

Pure3270 replaces the binary s3270 dependency in p3270 setups, eliminating the need for external installations (e.g., no compiling or downloading s3270 binaries).

Key Changes

  • Binary Replacement via Patching: Call pure3270.enable_replacement() before importing p3270. This monkey-patches p3270.P3270Client to delegate to pure3270's Session, handling connections, sends, and reads internally using standard asyncio instead of spawning s3270 processes.
  • Zero-Config Opt-In: No changes to your p3270 code required. The patching is global by default but reversible.
  • Handling Mismatches:
    • If p3270 version doesn't match (e.g., !=0.1.6, as checked in patches), logs a warning and skips patches gracefully (no error unless strict_version=True).
    • If p3270 is not installed, patching simulates with mocks and logs a warning; use standalone pure3270.Session instead.
    • Protocol differences: Pure3270 uses pure Python telnet/SSL, so ensure hosts support TN3270/TN3270E (RFC 1576/2355). SSL uses Python's ssl module.

Before / After

Before (with s3270):

  • Install s3270 binary.
  • import p3270; session = p3270.P3270Client(); session.connect(...) (spawns s3270).

After (with pure3270):

  • Install pure3270 as above.
  • import pure3270; pure3270.enable_replacement(); import p3270; session = p3270.P3270Client(); session.connect(...) (uses pure Python emulation).

Test migration by checking logs for "Patched Session ..." messages. For standalone scripts, switch to from pure3270 import Session.

Examples

See the examples/ directory for practical scripts:

Run them in your activated venv: python examples/example_patching.py. Replace mock hosts with real TN3270 servers (e.g., IBM z/OS systems) for production.

Troubleshooting

  • Venv Activation Issues: Ensure the venv is activated (prompt shows (.venv)). On Windows, use Scripts\activate.bat. If pip installs globally, recreate the venv.
  • Patching Fails: Check logs for version mismatches (e.g., p3270 !=0.1.6). Set strict_version=True to raise errors. If p3270 absent, use standalone mode.
  • Connection/Protocol Errors: Verify host/port (default 23/992 for SSL). Enable DEBUG logging: pure3270.setup_logging('DEBUG'). Common: Host doesn't support TN3270; test with tools like tn3270 client.
  • Screen Read Issues: Ensure read() is called after send(). For empty screens, check if BIND negotiation succeeded (logs show).
  • Async/Sync Mix: Use Session for sync code; AsyncSession for async. Don't mix in the same script without asyncio.run().

For more, enable verbose logging or consult architecture.md.

Credits

Credits: Some tests and examples in this project are inspired by and adapted from the IBM s3270 terminal emulator project, which served as a valuable reference for 3270 protocol handling and emulation techniques.

License and Contributing

See setup.py for author info. Contributions welcome via issues/PRs.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

pure3270-0.3.1.tar.gz (79.2 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

pure3270-0.3.1-py3-none-any.whl (88.8 kB view details)

Uploaded Python 3

File details

Details for the file pure3270-0.3.1.tar.gz.

File metadata

  • Download URL: pure3270-0.3.1.tar.gz
  • Upload date:
  • Size: 79.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for pure3270-0.3.1.tar.gz
Algorithm Hash digest
SHA256 835c633c016af11c2c9e6c293d8d2744a8e4a0d2d664310c97c776fdf3e7632a
MD5 9e94af6acc67fc991f835dc811daaa43
BLAKE2b-256 721211d330bd30531ef79685958e2ae3d8f412f921d2f351237df8d46e8ac010

See more details on using hashes here.

Provenance

The following attestation bundles were made for pure3270-0.3.1.tar.gz:

Publisher: ci.yml on dtg01100/pure3270

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pure3270-0.3.1-py3-none-any.whl.

File metadata

  • Download URL: pure3270-0.3.1-py3-none-any.whl
  • Upload date:
  • Size: 88.8 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for pure3270-0.3.1-py3-none-any.whl
Algorithm Hash digest
SHA256 26d9333132bfe20d5c14c009ba82fe3a2cbfcff8c014cb7b831a1df6519c0600
MD5 531710910ebcce7ce1272cbf56ec1d29
BLAKE2b-256 70283ad02b9f390620e627847b1021eceecd4261eb17b23856ba41db7e2b9f5a

See more details on using hashes here.

Provenance

The following attestation bundles were made for pure3270-0.3.1-py3-none-any.whl:

Publisher: ci.yml on dtg01100/pure3270

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page