Skip to main content

Batteries-included distribution of the PyNeedle runtime.

Project description

PyNeedle: The Semantic Pointer Runtime

License

A modern toolkit for decoupling meaning from implementation in Python applications.

English | 中文


What is PyNeedle?

PyNeedle is a small, powerful library designed to manage application strings, internationalization (i18n), and other addressable resources through a clean, intuitive, and type-safe API. It replaces "magic strings" with Semantic Pointers—objects that represent the meaning of a resource, not its value.

Think of it as finding a "needle" in the "haystack" of your application's resources, but instead of searching by a string key, you use a structured, code-like pointer.

The core of the library is the global L object (short for "Location" or "Lexicon"). Instead of writing this:

# Prone to typos, hard to refactor, no autocompletion
get_message("error.login.invalid_password")

You write this:

# Fluent, autocompletes, refactorable, type-safe
nexus.get(L.error.login.invalid_password)

Core Concepts

PyNeedle's architecture is simple and composed of three main parts:

  1. Semantic Pointer (L): An immutable object that represents a path in a logical "semantic universe". It's created fluently using attribute access (L.auth.login) or path-like joins (L.auth / "login"). It acts as the universal key for all resources.

  2. Resource Loader: A component responsible for loading data from a source. PyNeedle includes a FileSystemLoader for discovering .json files in your project and a MemoryLoader for tests or dynamic data. The contract is simple, so you can easily write your own loader for databases, APIs, etc.

  3. Nexus: The central runtime hub. The Nexus takes a list of loaders and resolves Semantic Pointers into their final string values. It intelligently handles language fallbacks and overlays, allowing you to merge resources from multiple sources with a clear priority order.

The pyneedle package provides a "batteries-included" global nexus instance that is pre-configured with a FileSystemLoader, making it incredibly easy to get started.

Key Features

  • ✨ Fluent & Expressive API: Create pointers naturally with L.user.profile.title.
  • ➕ Pointer Algebra: Perform powerful operations on pointers and sets of pointers, like distribution (L.user * {"name", "age"}) and concatenation ({L.a, L.b} / "end").
  • 🌐 Built-in I18n: The Nexus has first-class support for language resolution, environment variable detection (NEEDLE_LANG), and graceful fallbacks.
  • 📚 Layered Configuration: The OverlayNexus uses a ChainMap strategy to logically overlay resource files. A project/.stitcher/needle/en/override.json can easily override values from a library/needle/en/defaults.json without merging files.
  • 🔌 Extensible: Easily add custom ResourceLoaderProtocol implementations to fetch data from any source.
  • 📦 Decoupled Architecture: The project is split into logical packages (pyneedle-spec, pyneedle-pointer, pyneedle-nexus, pyneedle-runtime), promoting clean design and maintainability.
  • 🔒 Type Safe: When used with its companion tool, Stitcher, PyNeedle enables full static analysis and autocompletion for all your resources.

Quick Start

This example demonstrates the core functionality using an in-memory loader.

from needle import L, nexus
from needle.nexus import OverlayNexus, MemoryLoader

# 1. Define some resource data for different languages
resource_data = {
    "en": {
        "app.title": "My Awesome App",
        "user.greeting": "Welcome, {name}!",
    },
    "zh": {
        "app.title": "我的超棒应用",
    },
}

# 2. Create a loader with this data
memory_loader = MemoryLoader(data=resource_data)

# 3. Create a Nexus instance with the loader
#    (In a real app, you might just use the global 'nexus' instance)
local_nexus = OverlayNexus(loaders=[memory_loader], default_lang="en")

# 4. Resolve pointers to strings
# --- Resolving in the default language (en) ---
title_en = local_nexus.get(L.app.title)
print(f"English Title: {title_en}")
# Output: English Title: My Awesome App

# --- Explicitly requesting a language that exists ---
title_zh = local_nexus.get(L.app.title, lang="zh")
print(f"Chinese Title: {title_zh}")
# Output: Chinese Title: 我的超棒应用

# --- Requesting a key that falls back to the default language ---
# 'user.greeting' doesn't exist in 'zh', so it falls back to 'en'
greeting_fallback = local_nexus.get(L.user.greeting, lang="zh")
print(f"Fallback Greeting: {greeting_fallback}")
# Output: Fallback Greeting: Welcome, {name}!

# --- Requesting a key that doesn't exist anywhere ---
# It gracefully falls back to its own string representation (Identity Fallback)
non_existent = local_nexus.get(L.app.non_existent_key)
print(f"Non-existent Key: {non_existent}")
# Output: Non-existent Key: app.non_existent_key

Advanced Usage

Pointer Algebra

Create sets of pointers for powerful, expressive operations. PyNeedle supports fluent broadcasting and multi-indexing.

from needle import L

# 1. Multi-indexing shortcut
# Create a PointerSet effortlessly
user_fields = L.user['name', 'email']
# Result: PointerSet({L.user.name, L.user.email})

# 2. Attribute & Index Broadcasting
# Broadcast a suffix or index across the entire set fluently
form_labels = user_fields.label
# Result: PointerSet({L.user.name.label, L.user.email.label})

# 3. Cartesian Product Expansion
# Use multiplication to expand a pointer into a set
actions = {"read", "write"}
permissions = L.auth.user * actions
# Result: PointerSet({L.auth.user.read, L.auth.user.write})

# 4. Advanced Chaining
# Combine everything for powerful resource addressing
errors = L.api['v1', 'v2'][404].message
# Result: PointerSet({L.api.v1[404].message, L.api.v2[404].message})

File-Based Loading

The default global nexus instance uses a FileSystemLoader. Simply create the following directory structure in your project root:

my_project/
├── needle/
│   ├── en/
│   │   └── main.json
│   └── zh/
│       └── main.json
├── pyproject.toml
└── src/
    └── ...

needle/en/main.json:

{
    "app.title": "My App from File"
}

Now, your application code is incredibly simple:

# src/my_project/main.py
from needle import L, nexus

def main():
    # The global nexus automatically finds and loads the files
    print(nexus.get(L.app.title))

# Output: My App from File

Installation

Install the package from PyPI:

pip install pyneedle

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

pyneedle-0.1.5.tar.gz (9.0 kB view details)

Uploaded Source

Built Distribution

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

pyneedle-0.1.5-py3-none-any.whl (4.8 kB view details)

Uploaded Python 3

File details

Details for the file pyneedle-0.1.5.tar.gz.

File metadata

  • Download URL: pyneedle-0.1.5.tar.gz
  • Upload date:
  • Size: 9.0 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.11.14

File hashes

Hashes for pyneedle-0.1.5.tar.gz
Algorithm Hash digest
SHA256 cd50a67ca3ac907151958537f7a74d77eb3b5991dec58785e1d20b7048d8a055
MD5 fc72def1b0ea8bcfcd01b197ef59c5db
BLAKE2b-256 7135c6cd4879eb81d8b010c44ac62280936e52ca12a392285e5e50c8e0f700d7

See more details on using hashes here.

File details

Details for the file pyneedle-0.1.5-py3-none-any.whl.

File metadata

  • Download URL: pyneedle-0.1.5-py3-none-any.whl
  • Upload date:
  • Size: 4.8 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.11.14

File hashes

Hashes for pyneedle-0.1.5-py3-none-any.whl
Algorithm Hash digest
SHA256 83a16c3393824ca940f239ae5b1484665e777da3a19d0c06e4a884f37e5c7ef7
MD5 5b37b90927a4c897a893276dc2b61dd5
BLAKE2b-256 fd2616a1727d6e5bbfbeb03cb6feee2e9b2d5add7f624676e6c1f5a041a2b550

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