Skip to main content

Universal tracker interface, connector SDK, and sync engine for Spec Kitty

Project description

spec-kitty-tracker

Shared task-tracker abstraction layer for Spec Kitty CLI and SaaS.

Scope

  1. Canonical issue model (CanonicalIssue, ExternalRef, CanonicalLink)
  2. Tracker connector protocol with capability negotiation
  3. Doctrine-style source-of-truth ownership policies
  4. Conflict resolution and deterministic sync engine
  5. Bidirectional mission/issue sync bridge with decision reference traceability
  6. Connector registry and vendor connector implementations

Included Connectors

SaaS-Backed (via create_hosted_connector)

  1. LinearConnector
  2. JiraConnector
  3. GitHubConnector
  4. GitLabConnector

Out of Scope for Hosted Transport (this release)

  1. AzureDevOpsConnector

Local/Native (direct construction)

  1. BeadsConnector — local-first bd CLI adapter
  2. FPConnector — local-first fp CLI adapter

Test/Reference

  1. InMemoryConnector — fully functional reference connector

Installation

pip install -e .

For development tools:

pip install -e ".[dev]"

Quick Example

The primary integration path for SaaS-backed providers (Linear, Jira, GitHub, GitLab) uses create_hosted_connector with host-provided transport context. See Local/Native and Test Connectors for direct construction.

import asyncio
from spec_kitty_tracker import (
    create_hosted_connector,
    HostedConnectorRequest,
    LinearHostedParams,
    NangoConnectionContext,
    InMemoryIssueStore,
    OwnershipPolicy,
    OwnershipMode,
    SyncEngine,
)


async def main() -> None:
    # SaaS host provides identity context per operation
    nango_ctx = NangoConnectionContext(
        connection_id="user-connection-id",
        provider_config_key="linear",
        nango_secret_key="nango-secret",
    )

    # Factory constructs a connector routed through SaaS-owned transport
    connector = create_hosted_connector(HostedConnectorRequest(
        provider="linear",
        nango_context=nango_ctx,
        params=LinearHostedParams(team_id="TEAM-UUID"),
    ))

    store = InMemoryIssueStore()
    policy = OwnershipPolicy(mode=OwnershipMode.EXTERNAL_AUTHORITATIVE)
    engine = SyncEngine(connector=connector, store=store, policy=policy)

    async with connector:
        await engine.pull()


asyncio.run(main())

Hosted Discovery

The canonical path for downstream hosts to discover provider workspaces and bindable resources is the function-based public API: discover_workspaces and discover_resources. Both are exported from the top-level spec_kitty_tracker package and return typed dataclasses.

import asyncio

from spec_kitty_tracker import (
    create_hosted_connector,
    discover_resources,
    discover_workspaces,
    ConnectorRequestError,
    DiscoveryContractError,
    HostedConnectorRequest,
    NangoConnectionContext,
)
from spec_kitty_tracker.discovery.hosted_adapter import (
    connector_params_to_hosted_params,
)


async def integrate_linear() -> None:
    # 1. Build the per-user transport context. The host owns identity.
    nango_ctx = NangoConnectionContext(
        connection_id="user-connection-id",
        provider_config_key="linear",
        nango_secret_key="nango-secret",
    )

    # 2. Discover workspaces. Catch contract failures separately.
    try:
        workspaces = await discover_workspaces("linear", nango_ctx)
    except DiscoveryContractError as exc:
        # Provider returned malformed metadata — alert operators.
        print(f"contract violation: {exc.provider} {exc.field_path}")
        return
    except ConnectorRequestError as exc:
        # Network / auth / API error — retryable based on failure_class.
        print(f"request failed: {exc}")
        return

    workspace = workspaces.items[0]
    print(f"id={workspace.id} display={workspace.display}")

    # Read normalized optional metadata defensively.
    metadata = workspace.provider_context or {}
    handle = metadata.get("workspace_handle")  # str | None
    url = metadata.get("workspace_url")        # str | None

    # 3. Discover bindable resources within the chosen workspace.
    resources = await discover_resources("linear", workspace, nango_ctx)
    resource = resources.items[0]

    routing = resource.routing_metadata
    display_key = routing.get("display_key")     # str | None
    resource_url = routing.get("resource_url")   # str | None

    # 4. Translate connector_params to typed HostedParams + construct connector.
    hosted_params = connector_params_to_hosted_params(
        provider=resource.provider,
        connector_params=resource.connector_params,
    )
    connector = create_hosted_connector(HostedConnectorRequest(
        provider=resource.provider,
        nango_context=nango_ctx,
        params=hosted_params,
    ))

    async with connector:
        ...  # Use the connector with SyncEngine, mission_seed_from_issue, etc.


asyncio.run(integrate_linear())

The shape above is identical for linear, jira, github, and gitlab — only the value of the provider argument and the upstream Nango configuration change.

Hybrid metadata contract

Discovery results carry a hybrid metadata schema. Top-level dataclass fields (id, name, display, kind, provider, stable_ref, display_name, etc.) are the canonical source of truth for IDs and labels — they are always present and never duplicated into metadata. The two metadata containers (provider_context on workspaces and routing_metadata on resources) carry provider-native keys plus four optional normalized keys:

Container Key Type Semantic
provider_context workspace_handle str | None Short URL-safe slug for the workspace
provider_context workspace_url str | None Browser URL pointing to the workspace home
routing_metadata display_key str | None Short human-readable resource identifier
routing_metadata resource_url str | None Browser URL pointing to the resource home

These keys are optional: when a provider has no meaningful value, the key is absent. When present, the type is enforced at runtime by discover_workspaces / discover_resources — a malformed payload raises DiscoveryContractError. Hosts that want generic, cross-provider behavior should prefer top-level fields and the four normalized keys over provider-native extras.

See kitty-specs/006-hosted-discovery-contract-hardening/contracts/discovery-contract.md for the full contract specification, data-model.md for the per-provider expected-keys table, and quickstart.md for the full host integration walkthrough.

Deprecated shim modules

The modules spec_kitty_tracker.workspace_discovery and spec_kitty_tracker.resource_discovery are compatibility-only shims. They re-export the canonical symbols (discover_workspaces, discover_resources, DiscoveredWorkspace, DiscoveredResource, DiscoveryResult) for backward compatibility with older pinned downstream consumers, but they emit a DeprecationWarning on import. New code should import from the top-level package:

# Deprecated — emits DeprecationWarning
from spec_kitty_tracker.workspace_discovery import discover_workspaces
from spec_kitty_tracker.resource_discovery import discover_resources

# Canonical
from spec_kitty_tracker import discover_workspaces, discover_resources

The shim modules will not be removed in a backward-incompatible way without a separate migration mission.

Local/Native and Test Connectors

For connectors that do not use SaaS-hosted transport, construct them directly.

Local/Native (Beads)

from spec_kitty_tracker import BeadsConnector, BeadsConnectorConfig

config = BeadsConnectorConfig(workspace="my-project", cwd="/path/to/beads")
connector = BeadsConnector(config)

Test/Reference (InMemory)

from spec_kitty_tracker import InMemoryConnector

connector = InMemoryConnector(name="test", workspace="test-ws")

Migration/Advanced (direct SaaS-backed, non-product path)

# For test, migration, or advanced SDK use cases only.
# The Spec Kitty CLI/SaaS product path should use create_hosted_connector().
from spec_kitty_tracker import LinearConnector, LinearConnectorConfig

config = LinearConnectorConfig(api_key="test-key", team_id="TEAM-1")
connector = LinearConnector(config)

Design Notes

  1. The core contract is intentionally tracker-agnostic.
  2. OwnershipPolicy makes source-of-truth behavior explicit and auditable.
  3. SyncEngine supports pull, push, and bidirectional sync with conflict records.
  4. Capability flags gate optional features (e.g., hierarchy, dependencies, webhooks).
  5. Mission updates are idempotent and persist decision references back to source issues.

Docs

  1. Architecture
  2. Connector Contract
  3. Doctrine Policy Modes
  4. P0 Provider Matrix
  5. WP04 Contract Alignment Notes
  6. WP11 Sync Core Notes

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

spec_kitty_tracker-0.4.3.tar.gz (81.8 kB view details)

Uploaded Source

Built Distribution

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

spec_kitty_tracker-0.4.3-py3-none-any.whl (66.5 kB view details)

Uploaded Python 3

File details

Details for the file spec_kitty_tracker-0.4.3.tar.gz.

File metadata

  • Download URL: spec_kitty_tracker-0.4.3.tar.gz
  • Upload date:
  • Size: 81.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for spec_kitty_tracker-0.4.3.tar.gz
Algorithm Hash digest
SHA256 c7086ad782164ba069663b2a8f78972834b056ce99985be2d7b4f8245cf9e5f2
MD5 c08513d6c7812248ce7de11c9e74ffce
BLAKE2b-256 391cf3bd23a116b65874a0f4a360dc3249dad4b78c6ee9384a75e099b5b69b3c

See more details on using hashes here.

Provenance

The following attestation bundles were made for spec_kitty_tracker-0.4.3.tar.gz:

Publisher: publish-pypi.yml on Priivacy-ai/spec-kitty-tracker

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

File details

Details for the file spec_kitty_tracker-0.4.3-py3-none-any.whl.

File metadata

File hashes

Hashes for spec_kitty_tracker-0.4.3-py3-none-any.whl
Algorithm Hash digest
SHA256 2c393e767af33fa9ff3aaa98b1d97648966712ebcdf8fce1f6178548eacb4810
MD5 95764692587e518c186fa77c9ffb6800
BLAKE2b-256 0e0fca2ca110757e851bafc783ad2528a33bdf8a11ad094ce50b252a37ce16ca

See more details on using hashes here.

Provenance

The following attestation bundles were made for spec_kitty_tracker-0.4.3-py3-none-any.whl:

Publisher: publish-pypi.yml on Priivacy-ai/spec-kitty-tracker

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