#kafka #header

app miffy

A shadow-testing proxy: Send requests to a "reference" implementation, send the request to a "candidate" implementation, always respond with the "reference" implementation and log/publish both responses if they are not equal

2 releases (1 stable)

Uses new Rust 2024

new 1.0.0 Apr 1, 2025
0.1.0 Apr 1, 2025

#113 in HTTP server

MIT/Apache

48KB
1K SLoC

Miffy

My diffy.

A shadow-testing proxy: Send requests to a "reference" implementation, send the request to a "candidate" implementation, always respond with the "reference" implementation and log/publish both responses if they are not equal.

Design goals

  • reference always wins: if the candidate fails, is slow, not available or whatever it should not impact the reference, always return a response as fast as possible
  • offload complex comparison: only basic comparison, in case of doubt publish both responses as different to kafka

Development

Required tools:

Native libraries (required for rdakafka)

  • libssl-dev
  • libsasl2-dev

Demo

  • prepare the config: copy config.sample.toml to config.toml
  • start the demo-servers: cargo run --example demo. This will start two servers listening to localhost:3000 (the reference) and localhost:3001 (the candidate) with one endpoint: /api/{value}.
  • start kafka: docker-compose up -d
  • start miffy: cargo run.
  • send a request to a path under test: curl http://localhost:8080/api/3
  • send a request to any other path: curl http://localhost:8080
  • observe results in kafka: kcat -b localhost:9092 -e -t miffy

Configuration

Miffy looks for a file config.toml in it's working-directory, but you can point to a specific file via MIFFY_CONFIG-env-var.

  • get a working config by copying config.sample.toml to config.toml.
  • see config.default.toml for an explanation of different values and defaults.
  • all config-values values may be overriden via env-variable prefixed with MIFFY_.

Miffy uses rdkafka internally and allows to set all its properties, in config.toml section [kafka]. Since those values typically need to be set via environment variable (e.g. password for kafka, sasl.password), they may be set/overriden via KAFKA_SASL_PASSWORD etc.

Deployment

Miffy provides a separate management-port (default: 9000).

Currently, only a health-endpoint /healthz is available.

Headers

Sometimes services need to know if they take part in shadow-testing and also which role they play. E.g. "candidates" could still run in some dry-run mode where they not execute any side effects etc.

Miffy therefore always sends a header name X-Shadow-Test-Role:

  • candidate — the service is the candidate for the current request
  • reference — the service is the reference for the current request
  • upstream — there is no experiment configured for the current request/route, so the service is just used as upstream

Apart from that, miffy does not touch/change/add/remove any headers.

Benchmarking

To estimate the rough overhead added by miffy, there are just recipes to run wrk.

  • increase the open-file-limit for the shell running the demo: ulimit -n 8182
  • start the demo-servers: cargo run --example demo --release
  • start kafka: docker-compose up -d
  • increase open-file limit for the shell running miffy: ulimit -n 8182
  • start miffy (without access-logging): RUST_LOG=warn cargo run --release.
  • run the benchmark: just bench

License

Licensed under either of

at your option.

Contribution

Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in the work by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.

Dependencies

~27–41MB
~626K SLoC