Skip to main content

A lightweight, embeddable, pure-Python Redis-compatible server for testing.

Project description

RESP Server - Embeddable Redis-Compatible Server for Python

PyPI version Python 3.9+ CI License: MIT

A lightweight, pure-Python implementation of a Redis-compatible server, designed primarily for local development and unit testing.

[!NOTE] Use Case: This is an embedded server (like SQLite) for your tests. It complements redis-py (the client) by giving you a zero-dependency server to connect to without Docker.

Features

  • Minimal Dependencies: Uses click for CLI (pure-Python standard library otherwise).
  • Embeddable: Run it inside your pytest suite without Docker or external Redis installation.
  • RESP Compatible: Works with any Redis client library (redis-py, node-redis, go-redis, etc.).
  • Lite: Supports Strings, Lists, Streams, Pub/Sub, and Expiration (Lazy).

Installation

pip install resp-server

Quick Start

1. Run as a Standalone Server

# Starts server on port 6379
resp-server --port 6379

2. Embed in Python Tests (Pytest Example)

You can spin up the server programmatically in your tests/fixtures:

import pytest
import threading
from resp_server.core.server import Server
import redis

@pytest.fixture(scope="session")
def redis_server():
    # Start server in a background thread
    server = Server(port=6399)
    t = threading.Thread(target=server.start, daemon=True)
    t.start()
    yield server
    server.stop()

def test_my_app(redis_server):
    # Connect using standard client
    r = redis.Redis(port=6399, decode_responses=True)
    r.set("foo", "bar")
    assert r.get("foo") == "bar"

Architecture

Directory Structure

resp_server/
├── main.py                      # CLI Entry point
├── core/
│   ├── server.py                # Server Class & TCP logic
│   ├── command_execution.py     # Command Router
│   ├── datastore.py             # In-Memory DB
├── protocol/
│   ├── resp.py                  # RESP Parser

Codebase Structure & Module Interaction

graph TD
    %% Main Entry Point
    Main["resp_server/main.py<br/>(CLI Entry)"]
    
    %% Core Modules
    Server["resp_server/core/server.py<br/>(TCP Server Class)"]
    CmdExec["resp_server/core/command_execution.py<br/>(Command Router)"]
    Datastore["resp_server/core/datastore.py<br/>(In-Memory DB)"]
    
    %% Utilities
    RespProto["resp_server/protocol/resp.py<br/>(RESP Parser)"]

    %% Flow Relationships
    Main -->|1. Parses Args & Init| Server
    Server -->|2. Accepts Connections| CmdExec
    
    CmdExec -->|3. Parses Raw Bytes| RespProto
    CmdExec -->|4. Read/Write Data| Datastore

Features & Internals

This section explains what each feature is, how it works internally, and how to use it.

1. Redis Serialization Protocol (RESP)

  • What it is: The binary-safe protocol Redis uses to communicate.
  • How it works: The RESP Parser (resp_server/protocol/resp.py) reads bytes from the TCP socket, identifying types by their first byte (+ for strings, $ for bulk strings, * for arrays). It recursively parses nested arrays.
  • Usage: Transparent to the user. All clients (redis-py, node-redis, etc.) speak this automatically.

2. Strings & Expiration

  • What it is: Basic key-value storage with Time-To-Live (TTL).
  • How it works: Values are stored in DATA_STORE wrapper dicts: {'value': ..., 'expiry': timestamp}. The get_data_entry function checks the timestamp on every access (lazy expiry) and deletes the key if expired.
  • Usage:
    r.set("mykey", "Hello World", px=5000)  # Set with 5s expiry
    r.get("mykey")  # Returns "Hello World"
    

3. Lists

  • What it is: Ordered collections of strings.
  • How it works: Stored as Python lists in DATA_STORE. Supports push/pop operations from both ends.
  • Usage:
    r.rpush("mylist", "A", "B", "C")
    r.lrange("mylist", 0, -1)  # Returns ['A', 'B', 'C']
    

4. Streams

  • What it is: An append-only log data structure.
  • How it works: Stored in STREAMS as a list of entries. XADD validates IDs (must be incremental). XREAD supports blocking by using a threading.Condition variable to put the client thread to sleep until new data arrives.
  • Usage:
    r.xadd("mystream", {"sensor-id": "1234", "temperature": "19.8"})
    r.xrange("mystream", "-", "+")
    

5. Pub/Sub

  • What it is: Real-time messaging where publishers send messages to channels and subscribers receive them.
  • How it works:
    • CHANNEL_SUBSCRIBERS maps channel names to a Set of client sockets.
    • When PUBLISH is called, the server iterates through the socket list for that channel and writes the message directly to them.
  • Usage:
    # Client A
    p = r.pubsub()
    p.subscribe("mychannel")
    
    # Client B
    r.publish("mychannel", "Hello Subscribers!")
    

6. Utility Commands

  • PING: Returns PONG. Used to test connection health.
  • ECHO: Returns the given string. Useful for debugging connectivity.
  • TYPE: Returns the type of value stored at a key (string, list, stream, etc.).
  • KEYS: Returns a list of keys matching a pattern (only * wildcard supported).
  • CONFIG: Supports GET for retrieving server configuration (e.g., dir, dbfilename).

Production Readiness Assessment

⚠️ Current Status: Embedded / Development Use Only

This project is designed as an embedded server for local development and unit testing, similar to SQLite. It is Not Production Ready for critical business data.

Criteria Status Notes
Stability ✅ High Passes standard redis-py integration tests for supported features.
Concurrency ⚠️ Medium Uses threads (good for I/O), but limited by Python GIL for CPU-bound tasks.
Persistence ⚠️ Partial Loads RDB files but does not currently implement background saving (BGSAVE).
Security ❌ None No password authentication (AUTH) or TLS encryption implemented.
Scalability ❌ Low Single-threaded event loop (conceptually) via Python threads; not async/await based.

Usage Examples

Here are real sample outputs from running the server with redis-py:

Starting the Server

$ python3 -m resp_server.main --port 6399
RDB file not found at ./dump.rdb, starting with empty DATA_STORE.
Server: Starting server on localhost:6399...
Server: Listening for connections...

Basic Operations with redis-py

import redis
import time

r = redis.Redis(port=6399, decode_responses=True)

# Connection test
print(f'PING -> {r.ping()}')
# Output: PING -> True

# String operations
print(f'SET mykey "Hello World" -> {r.set("mykey", "Hello World")}')
# Output: SET mykey "Hello World" -> True

print(f'GET mykey -> {r.get("mykey")}')
# Output: GET mykey -> Hello World

# Expiration
print(f'SET temp "I expire in 2s" PX 2000 -> {r.set("temp", "I expire in 2s", px=2000)}')
# Output: SET temp "I expire in 2s" PX 2000 -> True

print(f'GET temp -> {r.get("temp")}')
# Output: GET temp -> I expire in 2s

time.sleep(2.1)
print(f'GET temp (after 2.1s) -> {r.get("temp")}')
# Output: GET temp (after 2.1s) -> None

# List operations
print(f'RPUSH mylist A B C -> {r.rpush("mylist", "A", "B", "C")}')
# Output: RPUSH mylist A B C -> 3

print(f'LRANGE mylist 0 -1 -> {r.lrange("mylist", 0, -1)}')
# Output: LRANGE mylist 0 -1 -> ['A', 'B', 'C']

Testing

Run the test suite to verify functionality:

# Install test dependencies
pip install pytest redis

# Run tests
pytest tests/

Sample Output:

============================= test session starts ==============================
collected 13 items

tests/test_datastore.py ........                                         [ 61%]
tests/test_integration.py ...                                            [ 84%]
tests/test_protocol.py ..                                                [100%]

============================== 13 passed in 2.45s ===============================

Project Origin

This project was built as part of the CodeCrafters Redis Challenge, extended with additional features for portfolio purposes.

License

MIT License - see LICENSE file for details.

Changelog

v0.2.3

  • Refactored CLI to use click library for robust argument parsing.

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

resp_server-0.2.3.tar.gz (24.5 kB view details)

Uploaded Source

Built Distribution

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

resp_server-0.2.3-py3-none-any.whl (24.6 kB view details)

Uploaded Python 3

File details

Details for the file resp_server-0.2.3.tar.gz.

File metadata

  • Download URL: resp_server-0.2.3.tar.gz
  • Upload date:
  • Size: 24.5 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.9.25

File hashes

Hashes for resp_server-0.2.3.tar.gz
Algorithm Hash digest
SHA256 5294293ea90252abb581ae6e056901ffe4d34085235a11ff8d5e3041c9dbf9d7
MD5 7abbe9390a9b0725a80e7bd2f5abaf93
BLAKE2b-256 a50cf6ae9514506372460ff78344447086680348b18f1eeaa9cbb95113bbf02c

See more details on using hashes here.

File details

Details for the file resp_server-0.2.3-py3-none-any.whl.

File metadata

  • Download URL: resp_server-0.2.3-py3-none-any.whl
  • Upload date:
  • Size: 24.6 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.9.25

File hashes

Hashes for resp_server-0.2.3-py3-none-any.whl
Algorithm Hash digest
SHA256 0fe39c24eea9a02e42f05b6220e4965ff4ac1f2ca03d3c3e326e61045cf80f20
MD5 e09369c4594a010629d5ddaba727b7b6
BLAKE2b-256 e6bf49e553e6f7b7d0c9fa41dce67eb91dd2e9d929b6c734774de63b2c5f1158

See more details on using hashes here.

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