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.1.tar.gz (81.5 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.1-py3-none-any.whl (66.4 kB view details)

Uploaded Python 3

File details

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

File metadata

  • Download URL: spec_kitty_tracker-0.4.1.tar.gz
  • Upload date:
  • Size: 81.5 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.1.tar.gz
Algorithm Hash digest
SHA256 e24e6923bf5194d5edb2c2a06898c2504b3cc226bea273271b3b61ecf5131383
MD5 6a01b6b4c73960006bd5d0f1b32a72d8
BLAKE2b-256 4e81e750a850027c38787766d7cbbe480987042452e37b74e6479147ede2398c

See more details on using hashes here.

Provenance

The following attestation bundles were made for spec_kitty_tracker-0.4.1.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.1-py3-none-any.whl.

File metadata

File hashes

Hashes for spec_kitty_tracker-0.4.1-py3-none-any.whl
Algorithm Hash digest
SHA256 cea8394d1edccadebf1f651d5ef36253e47f59ea2bb343226e3c8fbf4e6a2ae7
MD5 87fac15bfe93e6b8d840954340cfb319
BLAKE2b-256 931579be727c6ada91768f3ce76265f45f1ce53ad4fd5e3f5c558aa36480588e

See more details on using hashes here.

Provenance

The following attestation bundles were made for spec_kitty_tracker-0.4.1-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