1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188
//! # OpenTelemetry Propagator interface //! //! Propagators API consists of two main formats: //! //! - `BinaryFormat` is used to serialize and deserialize a value //! into a binary representation. //! - `HttpTextFormat` is used to inject and extract a value as //! text into carriers that travel in-band across process boundaries. //! //! Deserializing must set `is_remote` to true on the returned //! `SpanContext`. //! //! ## Binary Format //! //! `BinaryFormat` is a formatter to serialize and deserialize a value //! into a binary format. //! //! `BinaryFormat` MUST expose the APIs that serializes values into bytes, //! and deserializes values from bytes. //! //! ### ToBytes //! //! Serializes the given value into the on-the-wire representation. //! //! Required arguments: //! //! - the value to serialize, can be `SpanContext` or `DistributedContext`. //! //! Returns the on-the-wire byte representation of the value. //! //! ### FromBytes //! //! Creates a value from the given on-the-wire encoded representation. //! //! If the value could not be parsed, the underlying implementation //! SHOULD decide to return ether an empty value, an invalid value, or //! a valid value. //! //! Required arguments: //! //! - on-the-wire byte representation of the value. //! //! Returns a value deserialized from bytes. //! //! ## HTTP Text Format //! //! `HttpTextFormat` is a formatter that injects and extracts a value //! as text into carriers that travel in-band across process boundaries. //! //! Encoding is expected to conform to the HTTP Header Field semantics. //! Values are often encoded as RPC/HTTP request headers. //! //! The carrier of propagated data on both the client (injector) and //! server (extractor) side is usually an http request. Propagation is //! usually implemented via library-specific request interceptors, where //! the client-side injects values and the server-side extracts them. //! //! `HttpTextFormat` MUST expose the APIs that injects values into carriers, //! and extracts values from carriers. //! //! ### Fields //! //! The propagation fields defined. If your carrier is reused, you should //! delete the fields here before calling `inject`. //! //! For example, if the carrier is a single-use or immutable request object, //! you don't need to clear fields as they couldn't have been set before. //! If it is a mutable, retryable object, successive calls should clear //! these fields first. //! //! The use cases of this are: //! //! - allow pre-allocation of fields, especially in systems like gRPC //! Metadata //! - allow a single-pass over an iterator //! //! Returns list of fields that will be used by this formatter. //! //! ### Inject //! //! Injects the value downstream. For example, as http headers. //! //! Required arguments: //! //! - the `SpanContext` to be injected. //! - the carrier that holds propagation fields. For example, an outgoing //! message or http request. //! - the `Setter` invoked for each propagation key to add or remove. //! //! #### Setter argument //! //! Setter is an argument in `Inject` that puts value into given field. //! //! `Setter` allows a `HttpTextFormat` to set propagated fields into a //! carrier. //! //! `Setter` MUST be stateless and allowed to be saved as a constant to //! avoid runtime allocations. One of the ways to implement it is `Setter` //! class with `Put` method as described below. //! //! ##### Put //! //! Replaces a propagated field with the given value. //! //! Required arguments: //! //! - the carrier holds propagation fields. For example, an outgoing message //! or http request. //! - the key of the field. //! - the value of the field. //! //! The implementation SHOULD preserve casing (e.g. it should not transform //! `Content-Type` to `content-type`) if the used protocol is case insensitive, //! otherwise it MUST preserve casing. //! //! ### Extract //! //! Extracts the value from upstream. For example, as http headers. //! //! If the value could not be parsed, the underlying implementation will //! decide to return an object representing either an empty value, an invalid //! value, or a valid value. //! //! Required arguments: //! //! - the carrier holds propagation fields. For example, an outgoing message //! or http request. //! - the instance of `Getter` invoked for each propagation key to get. //! //! Returns the non-null extracted value. //! //! #### Getter argument //! //! Getter is an argument in `Extract` that get value from given field //! //! `Getter` allows a `HttpTextFormat` to read propagated fields from a //! carrier. //! //! `Getter` MUST be stateless and allowed to be saved as a constant to avoid //! runtime allocations. One of the ways to implement it is `Getter` class //! with `Get` method as described below. //! //! ##### Get //! //! The Get function MUST return the first value of the given propagation //! key or return `None` if the key doesn't exist. //! //! Required arguments: //! //! - the carrier of propagation fields, such as an HTTP request. //! - the key of the field. //! //! The `get` function is responsible for handling case sensitivity. If //! the getter is intended to work with an HTTP request object, the getter //! MUST be case insensitive. To improve compatibility with other text-based //! protocols, text format implementations MUST ensure to always use the //! canonical casing for their attributes. NOTE: Canonical casing for HTTP //! headers is usually title case (e.g. `Content-Type` instead of `content-type`). //! use crate::api; use std::collections::HashMap; #[cfg(feature = "base64")] pub mod base64_format; pub mod binary_propagator; pub mod composite_propagator; pub mod text_propagator; /// Carriers provide an interface for adding and removing fields from an /// underlying struct like `HashMap`. pub trait Carrier { /// Get a value for a key from the underlying data. fn get(&self, key: &'static str) -> Option<&str>; /// Add a key and value to the underlying. fn set(&mut self, key: &'static str, value: String); } impl<S: std::hash::BuildHasher> api::Carrier for HashMap<&'static str, String, S> { /// Get a value for a key from the HashMap. fn get(&self, key: &'static str) -> Option<&str> { self.get(key).map(|v| v.as_str()) } /// Set a key and value in the HashMap. fn set(&mut self, key: &'static str, value: String) { self.insert(key, value); } }