Process isolation for gevent applications — run any object in a clean subprocess, call methods transparently via ZMQ IPC.
Project description
gisolate
Gevent has hurt me a thousand times, yet I keep coming back for more. This library is proof of that love.
Process isolation for gevent applications. Run any object in a clean subprocess, call its methods transparently via ZMQ IPC.
Why
gevent's monkey.patch_all() replaces stdlib modules globally. Some libraries (database drivers, native async frameworks, etc.) break under monkey-patching. gisolate spawns a clean child process — no monkey-patching — and proxies method calls over ZMQ, so incompatible code runs in isolation while your gevent app stays cooperative.
Install
pip install gisolate
Requires Python 3.12+.
Quick Start
ProcessProxy — persistent child process
Proxy method calls to an object living in an isolated subprocess:
import gevent.monkey
gevent.monkey.patch_all()
from gisolate import ProcessProxy
# Define a factory (must be importable / picklable)
def create_client():
from some_native_lib import Client
return Client(host="localhost")
# Option 1: inline
proxy = ProcessProxy.create(create_client, timeout=30)
result = proxy.query("SELECT 1") # runs in child process
proxy.shutdown()
# Option 2: subclass
class ClientProxy(ProcessProxy):
client_factory = staticmethod(create_client)
timeout = 30
with ClientProxy() as proxy:
result = proxy.query("SELECT 1")
run_in_subprocess — one-shot call
Run a single function in a subprocess and get the result:
from gisolate import run_in_subprocess
def heavy_compute(n):
return sum(range(n))
result = run_in_subprocess(heavy_compute, args=(10_000_000,), timeout=60)
ProcessBridge — cross-process RPC
ZMQ-based RPC bridge for server/client architectures. Server side uses gevent, client side uses asyncio:
from gisolate import ProcessBridge
# Server (gevent side)
server = ProcessBridge("ipc:///tmp/rpc.sock", mode="server")
_ = server.address # starts listening
# Client (asyncio side)
import asyncio
async def main():
client = ProcessBridge("ipc:///tmp/rpc.sock", mode="client")
result = await client.call(lambda x, y: x + y, 3, 4, timeout=5)
print(result) # 7
client.close()
asyncio.run(main())
server.close()
ThreadLocalProxy — per-thread instances
Thread-local proxy using unpatched threading.local for true isolation in gevent.threadpool:
from gisolate import ThreadLocalProxy
proxy = ThreadLocalProxy(create_client)
proxy.query("SELECT 1") # each real OS thread gets its own instance
Child Process Modes
patch_kwargs |
Child process runtime |
|---|---|
None (default) |
asyncio event loop |
dict |
gevent with patch_all(**patch_kwargs) |
# Child uses asyncio (default)
proxy = ProcessProxy.create(factory)
# Child uses gevent with selective patching
proxy = ProcessProxy.create(factory, patch_kwargs={"thread": False, "os": False})
API Reference
ProcessProxy
ProcessProxy.create(factory, *, timeout=24, mp_context=None, patch_kwargs=None)— create a proxy without subclassingproxy.<method>(*args, **kwargs)— transparently call any method on the remote objectproxy.restart_process()— kill and restart child processproxy.shutdown()— gracefully stop child process- Supports context manager (
withstatement) - Thread-safe: usable from greenlets and native threads
run_in_subprocess(target, args=(), kwargs=None, *, timeout=3600, mp_context=None)
Run a function in an isolated subprocess. Blocks with gevent-safe polling.
ProcessBridge(address, mode=None)
bridge.address— IPC address (starts server if needed)await bridge.call(func, *args, timeout=60, **kwargs)— async RPC call (client mode)bridge.close()— cleanup resources
ThreadLocalProxy(factory)
Transparent proxy delegating attribute access to a per-thread instance.
spawn_on_main_hub(func, *args, **kwargs)
Schedule a function on the main gevent hub without waiting. Thread-safe, fire-and-forget.
ProcessError
Raised when a child process dies or communication fails.
RemoteError
Wrapper for exceptions from the child process that can't be pickled. Preserves the original exception type name and message.
shutdown_hub()
Explicitly stop the internal gevent hub loop. Registered via atexit automatically.
set_default_mp_context(ctx) / get_default_mp_context()
Configure the default multiprocessing context for all proxies (default: "spawn").
License
MIT
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 gisolate-0.1.5.tar.gz.
File metadata
- Download URL: gisolate-0.1.5.tar.gz
- Upload date:
- Size: 53.3 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
1641e15b136e987b51e58e15187cc5cb797eb537cbb4dd3f29eeacf1b18ca7b1
|
|
| MD5 |
0dd50cfcfeb016e39f142c5773fedf71
|
|
| BLAKE2b-256 |
1930ce9dc9120c9be3edd7105db6e6cd30407364503451b5ebb53cfa3deab8cd
|
File details
Details for the file gisolate-0.1.5-py3-none-any.whl.
File metadata
- Download URL: gisolate-0.1.5-py3-none-any.whl
- Upload date:
- Size: 17.5 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
7bc5062e052e1f67426cc21352dfada9a7204c5ca309effce5028c8520b25806
|
|
| MD5 |
560b180f9f150ce7dc21bb903e59610c
|
|
| BLAKE2b-256 |
29e3ef65663d3cecd87bf7285443d36d3ef5733f67864f6366a10dae5f92860c
|