The single, language-neutral set of cases every BabelQueue SDK must satisfy. It is the executable form of the wire contract — if two SDKs both pass this suite, a message one produces is consumable by the other.
Canonical source. Each SDK vendors a copy under
tests/conformance/(kept in sync from here). Published at https://babelqueue.com in human form.
| Path | What |
|---|---|
schema/message-envelope.schema.json |
The canonical envelope JSON Schema (draft-07). Validates producer output (job required). |
fixtures/*.json |
Canonical envelopes — one per case (pure envelopes, schema-validatable). |
manifest.json |
The case list: each case's fixture file, whether it's valid, and the expect values a consumer must derive. Drives a generic runner in any language. |
Decode each fixture with the SDK's core, then:
valid: true— the envelope is acceptable:accepts(envelope)is true;- the resolved URN equals
expect.urn(acceptingurnas an inbound alias forjob); dataequalsexpect.data;attemptsequalsexpect.attempts;meta.lang/meta.schema_versionequalexpect.lang/expect.schema_version;- if
expect.dead_letteris present, the message carries a matchingdead_letterblock.
valid: false— the SDK must reject it:accepts(envelope)is false (e.g. unknownmeta.schema_version, or nojob/urn).
Per-message fields (meta.id, trace_id, meta.created_at) are intrinsically
unique and are asserted for presence/shape, not value.
Note:
urn-alias.jsonusesurninstead ofjob. That is a consumer tolerance, not valid producer output — so it intentionally would NOT pass the producer JSON Schema (which requiresjob), but a consumer'saccepts()must still accept it.
The cases above lock the envelope (broker-agnostic). The sqs block locks the
Amazon SQS binding (broker-bindings.md §3) — the native
projection and reconciliation each SQS transport adds on top of the identical body.
Every SDK that ships an SQS transport must satisfy it:
sqs.attribute_projection— encodeattribute_projection.envelope_file, run the transport's produce-side projection, and assert the resulting nativeMessageAttributesequalattribute_projection.message_attributesexactly (same keys,DataTypeStringfor ids/strings andNumberfor counters, sameStringValue).sqs.attempts_reconciliation— for each case, the consume-side reconciliation MUST yieldexpected_attempts=max(body_attempts, ApproximateReceiveCount − 1): a first delivery reads0, an absent/garbage count is ignored, and a runtime-incremented count is never lowered. A drop-in driver that instead surfaces the broker's native count (e.g. Laravel'sSqsJob::attempts()=ApproximateReceiveCount) is exempt and documents that divergence.
Per-message attribute values reuse fixtures/order-created.json, so the expected
projection is deterministic. SDKs without an SQS transport ignore this block.
The asb block locks the Azure Service Bus binding
(broker-bindings.md §4) the same way. Every SDK that ships an
ASB transport must satisfy it:
asb.property_projection— encodeproperty_projection.envelope_file, run the transport's produce-side projection, and assert the native message fields equalproperty_projection.message(subject= the URN,correlation_id=trace_id,message_id=meta.id,content_type=application/json) and theApplicationPropertiesequalproperty_projection.application_propertiesexactly — as native AMQP-typed values (bq-schema-versionandbq-created-atstay numbers,bq-source-langa string; not theDataType-wrapped strings SQS uses).asb.attempts_reconciliation— for each case, the consume-side reconciliation MUST yieldexpected_attempts=max(body_attempts, delivery_count − 1): a first delivery (DeliveryCount1) reads0,DeliveryCount ≤ 1leaves the body's own count untouched, and a runtime-incremented count is never lowered. The rule is identical for the native-consumer SDKs (.NET/Java/Node, nativeAbandon) and the Transport+App SDKs (Python/Go, republish-retry).
The five ASB SDKs (babelqueue-dotnet-azureservicebus, babelqueue-java-azureservicebus,
@babelqueue/azure-service-bus, babelqueue Python AsbTransport,
babelqueue-go/azureservicebus) vendor this manifest via sync.sh; their conformance
runners are wired next. SDKs without an ASB transport ignore this block.
The pulsar block locks the Apache Pulsar binding
(broker-bindings.md §5) the same way. Every SDK that ships a
Pulsar transport must satisfy it:
pulsar.property_projection— encodeproperty_projection.envelope_file, run the transport's produce-side projection, and assert the native messagepropertiesequalproperty_projection.propertiesexactly. Pulsar properties are string→string, so every value is a string —bq-job= the URN,bq-trace-id=trace_id,bq-message-id=meta.id, and the stringifiedbq-schema-version("1"),bq-source-lang, andbq-attempts("0"). The payload stays the byte-identical envelope; the native publish time mirrorsmeta.created_at(broker-set, body authoritative — not asserted).pulsar.attempts_reconciliation— for each case, the consume-side reconciliation MUST yieldexpected_attempts=max(body_attempts, redelivery_count). Pulsar'sRedeliveryCountis 0-based (0 on first delivery) so it maps directly with no −1; a runtime-incremented count is never lowered, andredelivery_count0 leaves the body's own count untouched (the runtime retries by republishing with attempts+1, resetting the broker count to 0). The rule is identical for the native-consumer SDKs (.NET/Java/Node) and the Transport+App SDKs (Python/Go).
The five Pulsar SDKs (babelqueue-dotnet-pulsar, babelqueue-java-pulsar,
@babelqueue/pulsar, babelqueue Python PulsarTransport, babelqueue-go/pulsar) vendor
this manifest via sync.sh; their conformance runners are wired next. SDKs without a
Pulsar transport ignore this block.
The kafka block locks the Apache Kafka binding
(broker-bindings.md §6) the same way. Every SDK that ships a Kafka
transport must satisfy it:
kafka.property_projection— encodeproperty_projection.envelope_file, run the transport's produce-side projection, and assert the native recordheadersequalproperty_projection.headersexactly. Kafka headers are bytes, decoded as UTF-8 strings —bq-job= the URN,bq-trace-id=trace_id,bq-message-id=meta.id, and the stringifiedbq-schema-version("1"),bq-source-lang, andbq-attempts("0"). The record value stays the byte-identical envelope and the record timestamp mirrorsmeta.created_at(Unix ms).kafka.attempts_reconciliation— for each case, the consume-side reconciliation MUST yieldexpected_attempts= thebq-attemptsheader when present (authoritative — Kafka has no native delivery count), else the body's ownattempts(the fallback for a non-BabelQueue producer). This is not a max: the header overrides the body even when lower. Anullheader_attemptsmeans the header is absent. The rule is identical across all five Kafka SDKs.
The five Kafka SDKs (babelqueue-dotnet-kafka, babelqueue-java-kafka, @babelqueue/kafka,
babelqueue Python KafkaTransport, babelqueue-go/kafka) vendor this manifest via
sync.sh; their conformance runners are wired next. SDKs without a Kafka transport ignore
this block.
Each SDK ships a conformance test that loads manifest.json + fixtures/ from its
vendored tests/conformance/ copy and runs the envelope checks above:
- PHP:
vendor/bin/phpunit(theConformanceTest). - Python:
python -m unittest/pytest(test_conformance.py).
The sqs block is run by each SDK's SQS transport against the same vendored
manifest — wired + green in all six:
| SDK | Test | Reads | attribute_projection |
attempts_reconciliation |
|---|---|---|---|---|
| Go | babelqueue-go/sqs TestSqsConformance |
core's testdata/conformance/ |
✅ | ✅ |
| Python | babelqueue-python test_sqs_conformance.py |
tests/conformance/ |
✅ | ✅ |
| Node | @babelqueue/sqs sqs-conformance.test.ts |
test/conformance/ |
✅ | ✅ |
| Java | babelqueue-java-sqs SqsConformanceTest |
src/test/resources/conformance/ |
✅ | ✅ |
| .NET | babelqueue-dotnet-sqs SqsConformanceTests |
copied conformance/ |
✅ | ✅ |
| PHP | php-sdk SqsConformanceTest |
tests/conformance/ |
✅ | n/a¹ |
¹ php-sdk's SqsTransport is produce-only; the consume-side reconciliation lives in
the Laravel babelqueue-sqs driver, which surfaces the broker's native count (exempt per
the block's note). The three standalone transport repos (node-adapters, babelqueue-java-sqs,
babelqueue-dotnet-sqs) vendor their own copy via sync.sh (added to its targets).
conformance/ is the source of truth. Run ./sync.sh to copy schema/,
fixtures/ and manifest.json into each sibling SDK's vendored directory (paths
differ per SDK — Go testdata/, Java src/test/resources/, the rest
tests/conformance/).
Drift is guarded automatically:
./sync.sh --check— diffs every vendored copy against the canonical suite and exits non-zero on drift (the local counterpart to the CI guard). Run it before committing fixture changes.- Each core SDK's CI has a
conformancejob that shallow-clones this repo and diffs its vendored copy against the canonicalmanifest.json/fixtures//schema/— so a stale or hand-edited copy turns the SDK's build red. - This repo's CI validates the canonical suite itself (every fixture/schema is
valid JSON,
manifest.schema_versionis 1, and every case'sfileexists with the rightexpect/reasonblock).
So: edit a fixture here → ./sync.sh → commit each SDK. Forget to re-vendor and CI
catches it.