Pure Python 3270 emulator
Project description
Pure3270: Pure Python 3270 Terminal Emulation Library
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
AsyncSessionsupporting 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 patchp3270automatically. - Standalone usage: Use
SessionorAsyncSessiondirectly withoutp3270. - Pythonic API: Context managers, async support, and structured error handling.
- Compatibility: Mirrors
s3270andp3270interfaces 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:
- Install
p3270in your venv:pip install p3270. - 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/:
python-package.yml- Runs tests and linting across multiple Python versionsreports.yml- Generates and publishes coverage and linting reportspython-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:
- Fork the repository and create a feature branch.
- Install dev dependencies and run tests/linting locally.
- Make changes and add tests for new features.
- Ensure code passes
blackformatting andflake8linting. - 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 importingp3270. This monkey-patchesp3270.P3270Clientto delegate to pure3270'sSession, handling connections, sends, and reads internally using standard asyncio instead of spawnings3270processes. - Zero-Config Opt-In: No changes to your
p3270code required. The patching is global by default but reversible. - Handling Mismatches:
- If
p3270version doesn't match (e.g., !=0.1.6, as checked in patches), logs a warning and skips patches gracefully (no error unlessstrict_version=True). - If
p3270is not installed, patching simulates with mocks and logs a warning; use standalonepure3270.Sessioninstead. - Protocol differences: Pure3270 uses pure Python telnet/SSL, so ensure hosts support TN3270/TN3270E (RFC 1576/2355). SSL uses Python's
sslmodule.
- If
Before / After
Before (with s3270):
- Install
s3270binary. 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:
example_patching.py: Demonstrates applying patches and verifying redirection.example_end_to_end.py: Full p3270 usage after patching (with mock host).example_standalone.py: Direct pure3270 usage without p3270.
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, useScripts\activate.bat. Ifpipinstalls globally, recreate the venv. - Patching Fails: Check logs for version mismatches (e.g.,
p3270!=0.1.6). Setstrict_version=Trueto raise errors. Ifp3270absent, 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 liketn3270client. - Screen Read Issues: Ensure
read()is called aftersend(). For empty screens, check if BIND negotiation succeeded (logs show). - Async/Sync Mix: Use
Sessionfor sync code;AsyncSessionfor async. Don't mix in the same script withoutasyncio.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
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
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
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
835c633c016af11c2c9e6c293d8d2744a8e4a0d2d664310c97c776fdf3e7632a
|
|
| MD5 |
9e94af6acc67fc991f835dc811daaa43
|
|
| BLAKE2b-256 |
721211d330bd30531ef79685958e2ae3d8f412f921d2f351237df8d46e8ac010
|
Provenance
The following attestation bundles were made for pure3270-0.3.1.tar.gz:
Publisher:
ci.yml on dtg01100/pure3270
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pure3270-0.3.1.tar.gz -
Subject digest:
835c633c016af11c2c9e6c293d8d2744a8e4a0d2d664310c97c776fdf3e7632a - Sigstore transparency entry: 499687966
- Sigstore integration time:
-
Permalink:
dtg01100/pure3270@580399cb90184544e07394ba757a4b0d13119ecb -
Branch / Tag:
refs/tags/v0.3.1 - Owner: https://github.com/dtg01100
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
ci.yml@580399cb90184544e07394ba757a4b0d13119ecb -
Trigger Event:
release
-
Statement type:
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
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
26d9333132bfe20d5c14c009ba82fe3a2cbfcff8c014cb7b831a1df6519c0600
|
|
| MD5 |
531710910ebcce7ce1272cbf56ec1d29
|
|
| BLAKE2b-256 |
70283ad02b9f390620e627847b1021eceecd4261eb17b23856ba41db7e2b9f5a
|
Provenance
The following attestation bundles were made for pure3270-0.3.1-py3-none-any.whl:
Publisher:
ci.yml on dtg01100/pure3270
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pure3270-0.3.1-py3-none-any.whl -
Subject digest:
26d9333132bfe20d5c14c009ba82fe3a2cbfcff8c014cb7b831a1df6519c0600 - Sigstore transparency entry: 499687995
- Sigstore integration time:
-
Permalink:
dtg01100/pure3270@580399cb90184544e07394ba757a4b0d13119ecb -
Branch / Tag:
refs/tags/v0.3.1 - Owner: https://github.com/dtg01100
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
ci.yml@580399cb90184544e07394ba757a4b0d13119ecb -
Trigger Event:
release
-
Statement type: