From f53ea2bdfc7c84e7c553d319ff0a3558a7951820 Mon Sep 17 00:00:00 2001 From: Luca Palmieri Date: Tue, 17 May 2022 12:59:09 +0200 Subject: [PATCH 01/12] Use index_map::IndexMap instead of Vec to store SpanBuilder's attributes. --- opentelemetry-api/Cargo.toml | 1 + opentelemetry-api/src/trace/tracer.rs | 10 ++++------ opentelemetry-sdk/Cargo.toml | 3 ++- opentelemetry-sdk/src/trace/sampler.rs | 14 ++++++-------- opentelemetry-sdk/src/trace/tracer.rs | 15 ++++++++------- 5 files changed, 21 insertions(+), 22 deletions(-) diff --git a/opentelemetry-api/Cargo.toml b/opentelemetry-api/Cargo.toml index 257ad09a47..1406d9f5a5 100644 --- a/opentelemetry-api/Cargo.toml +++ b/opentelemetry-api/Cargo.toml @@ -11,6 +11,7 @@ lazy_static = "1.4" pin-project = { version = "1.0.2", optional = true } thiserror = "1" tokio-stream = { version = "0.1", optional = true } +indexmap = "1" [package.metadata.docs.rs] all-features = true diff --git a/opentelemetry-api/src/trace/tracer.rs b/opentelemetry-api/src/trace/tracer.rs index 1596d62b37..ccd533a3d2 100644 --- a/opentelemetry-api/src/trace/tracer.rs +++ b/opentelemetry-api/src/trace/tracer.rs @@ -1,9 +1,7 @@ -use crate::{ - trace::{Event, Link, Span, SpanId, SpanKind, Status, TraceContextExt, TraceId, TraceState}, - Context, KeyValue, -}; +use crate::{trace::{Event, Link, Span, SpanId, SpanKind, Status, TraceContextExt, TraceId, TraceState}, Context, KeyValue, Key, Value}; use std::borrow::Cow; use std::time::SystemTime; +use indexmap::IndexMap; /// The interface for constructing [`Span`]s. /// @@ -259,7 +257,7 @@ pub struct SpanBuilder { pub end_time: Option, /// Span attributes - pub attributes: Option>, + pub attributes: Option>, /// Span events pub events: Option>, @@ -325,7 +323,7 @@ impl SpanBuilder { } /// Assign span attributes - pub fn with_attributes(self, attributes: Vec) -> Self { + pub fn with_attributes(self, attributes: IndexMap) -> Self { SpanBuilder { attributes: Some(attributes), ..self diff --git a/opentelemetry-sdk/Cargo.toml b/opentelemetry-sdk/Cargo.toml index 7d810cbdfc..090c9b23c3 100644 --- a/opentelemetry-sdk/Cargo.toml +++ b/opentelemetry-sdk/Cargo.toml @@ -12,6 +12,7 @@ fnv = { version = "1.0", optional = true } futures-channel = "0.3" futures-executor = "0.3" futures-util = { version = "0.3", default-features = false, features = ["std", "sink", "async-await-macro"] } +indexmap = { version = "1", optional = true } lazy_static = "1.4" once_cell = "1.10" opentelemetry-api = { version = "0.1", path = "../opentelemetry-api/" } @@ -34,7 +35,7 @@ rand_distr = "0.4.0" [features] default = ["trace"] -trace = ["opentelemetry-api/trace", "crossbeam-channel", "rand", "pin-project", "async-trait", "percent-encoding"] +trace = ["opentelemetry-api/trace", "crossbeam-channel", "rand", "pin-project", "async-trait", "percent-encoding", "indexmap"] metrics = ["opentelemetry-api/metrics", "dashmap", "fnv"] testing = ["opentelemetry-api/testing", "trace", "metrics", "rt-async-std", "rt-tokio", "rt-tokio-current-thread", "tokio/macros", "tokio/rt-multi-thread"] rt-tokio = ["tokio", "tokio-stream"] diff --git a/opentelemetry-sdk/src/trace/sampler.rs b/opentelemetry-sdk/src/trace/sampler.rs index 8928632584..dff7d09628 100644 --- a/opentelemetry-sdk/src/trace/sampler.rs +++ b/opentelemetry-sdk/src/trace/sampler.rs @@ -38,13 +38,11 @@ //! MUST NOT allow this combination. use crate::InstrumentationLibrary; -use opentelemetry_api::{ - trace::{ - Link, SamplingDecision, SamplingResult, SpanKind, TraceContextExt, TraceId, TraceState, - }, - Context, KeyValue, -}; +use opentelemetry_api::{trace::{ + Link, SamplingDecision, SamplingResult, SpanKind, TraceContextExt, TraceId, TraceState, +}, Context, Value, Key}; use std::convert::TryInto; +use indexmap::IndexMap; /// The `ShouldSample` interface allows implementations to provide samplers /// which will return a sampling `SamplingResult` based on information that @@ -58,7 +56,7 @@ pub trait ShouldSample: Send + Sync + std::fmt::Debug { trace_id: TraceId, name: &str, span_kind: &SpanKind, - attributes: &[KeyValue], + attributes: &IndexMap, links: &[Link], instrumentation_library: &InstrumentationLibrary, ) -> SamplingResult; @@ -86,7 +84,7 @@ impl ShouldSample for Sampler { trace_id: TraceId, name: &str, span_kind: &SpanKind, - attributes: &[KeyValue], + attributes: &IndexMap, links: &[Link], instrumentation_library: &InstrumentationLibrary, ) -> SamplingResult { diff --git a/opentelemetry-sdk/src/trace/tracer.rs b/opentelemetry-sdk/src/trace/tracer.rs index ace67a34e9..b117ab332b 100644 --- a/opentelemetry-sdk/src/trace/tracer.rs +++ b/opentelemetry-sdk/src/trace/tracer.rs @@ -20,7 +20,8 @@ use opentelemetry_api::trace::{ Link, SamplingDecision, SamplingResult, SpanBuilder, SpanContext, SpanId, SpanKind, TraceContextExt, TraceFlags, TraceId, TraceState, }; -use opentelemetry_api::{Context, KeyValue}; +use opentelemetry_api::{Context, Key, KeyValue, Value}; +use indexmap::IndexMap; use std::fmt; use std::sync::Weak; @@ -72,7 +73,7 @@ impl Tracer { trace_id: TraceId, name: &str, span_kind: &SpanKind, - attributes: &[KeyValue], + attributes: &IndexMap, links: &[Link], config: &Config, instrumentation_library: &InstrumentationLibrary, @@ -218,14 +219,14 @@ impl opentelemetry_api::trace::Tracer for Tracer { } = builder; // Build optional inner context, `None` if not recording. - let mut span = if let Some((flags, mut extra_attrs, trace_state)) = sampling_decision { - if !extra_attrs.is_empty() { - attribute_options.append(&mut extra_attrs); + let mut span = if let Some((flags, extra_attrs, trace_state)) = sampling_decision { + for extra_attr in extra_attrs { + attribute_options.insert(extra_attr.key, extra_attr.value); } let mut attributes = EvictedHashMap::new(span_limits.max_attributes_per_span, attribute_options.len()); - for attribute in attribute_options { - attributes.insert(attribute); + for (key, value) in attribute_options { + attributes.insert(KeyValue::new(key, value)); } let mut links = EvictedQueue::new(span_limits.max_links_per_span); if let Some(link_options) = &mut link_options { From 0b12dbc8d3de24b6cfb95579b36d3959383d9642 Mon Sep 17 00:00:00 2001 From: Luca Palmieri Date: Tue, 17 May 2022 13:00:50 +0200 Subject: [PATCH 02/12] Test suite compiles. --- opentelemetry-sdk/src/trace/sampler.rs | 4 ++-- opentelemetry-sdk/src/trace/tracer.rs | 14 ++++++-------- 2 files changed, 8 insertions(+), 10 deletions(-) diff --git a/opentelemetry-sdk/src/trace/sampler.rs b/opentelemetry-sdk/src/trace/sampler.rs index dff7d09628..df0b806a49 100644 --- a/opentelemetry-sdk/src/trace/sampler.rs +++ b/opentelemetry-sdk/src/trace/sampler.rs @@ -240,7 +240,7 @@ mod tests { trace_id, name, &SpanKind::Internal, - &[], + &IndexMap::new(), &[], &InstrumentationLibrary::default(), ) @@ -282,7 +282,7 @@ mod tests { TraceId::from_u128(1), "should sample", &SpanKind::Internal, - &[], + &IndexMap::new(), &[], &instrumentation_library, ); diff --git a/opentelemetry-sdk/src/trace/tracer.rs b/opentelemetry-sdk/src/trace/tracer.rs index b117ab332b..8860217cd4 100644 --- a/opentelemetry-sdk/src/trace/tracer.rs +++ b/opentelemetry-sdk/src/trace/tracer.rs @@ -294,18 +294,16 @@ impl opentelemetry_api::trace::Tracer for Tracer { #[cfg(all(test, feature = "testing", feature = "trace"))] mod tests { + use indexmap::IndexMap; use crate::{ testing::trace::TestSpan, trace::{Config, Sampler, ShouldSample}, InstrumentationLibrary, }; - use opentelemetry_api::{ - trace::{ - Link, SamplingDecision, SamplingResult, Span, SpanContext, SpanId, SpanKind, - TraceContextExt, TraceFlags, TraceId, TraceState, Tracer, TracerProvider, - }, - Context, KeyValue, - }; + use opentelemetry_api::{trace::{ + Link, SamplingDecision, SamplingResult, Span, SpanContext, SpanId, SpanKind, + TraceContextExt, TraceFlags, TraceId, TraceState, Tracer, TracerProvider, + }, Context, Value, Key}; #[derive(Debug)] struct TestSampler {} @@ -317,7 +315,7 @@ mod tests { _trace_id: TraceId, _name: &str, _span_kind: &SpanKind, - _attributes: &[KeyValue], + _attributes: &IndexMap, _links: &[Link], _instrumentation_library: &InstrumentationLibrary, ) -> SamplingResult { From 3350ad529a16a05e044e7d394a615f40bcdbbce9 Mon Sep 17 00:00:00 2001 From: Luca Palmieri Date: Tue, 17 May 2022 15:05:25 +0200 Subject: [PATCH 03/12] Rustfmt --- opentelemetry-api/src/trace/tracer.rs | 7 +++++-- opentelemetry-sdk/src/trace/sampler.rs | 11 +++++++---- opentelemetry-sdk/src/trace/tracer.rs | 15 +++++++++------ 3 files changed, 21 insertions(+), 12 deletions(-) diff --git a/opentelemetry-api/src/trace/tracer.rs b/opentelemetry-api/src/trace/tracer.rs index ccd533a3d2..c90d3011ee 100644 --- a/opentelemetry-api/src/trace/tracer.rs +++ b/opentelemetry-api/src/trace/tracer.rs @@ -1,7 +1,10 @@ -use crate::{trace::{Event, Link, Span, SpanId, SpanKind, Status, TraceContextExt, TraceId, TraceState}, Context, KeyValue, Key, Value}; +use crate::{ + trace::{Event, Link, Span, SpanId, SpanKind, Status, TraceContextExt, TraceId, TraceState}, + Context, Key, KeyValue, Value, +}; +use indexmap::IndexMap; use std::borrow::Cow; use std::time::SystemTime; -use indexmap::IndexMap; /// The interface for constructing [`Span`]s. /// diff --git a/opentelemetry-sdk/src/trace/sampler.rs b/opentelemetry-sdk/src/trace/sampler.rs index df0b806a49..65efd03a63 100644 --- a/opentelemetry-sdk/src/trace/sampler.rs +++ b/opentelemetry-sdk/src/trace/sampler.rs @@ -38,11 +38,14 @@ //! MUST NOT allow this combination. use crate::InstrumentationLibrary; -use opentelemetry_api::{trace::{ - Link, SamplingDecision, SamplingResult, SpanKind, TraceContextExt, TraceId, TraceState, -}, Context, Value, Key}; -use std::convert::TryInto; use indexmap::IndexMap; +use opentelemetry_api::{ + trace::{ + Link, SamplingDecision, SamplingResult, SpanKind, TraceContextExt, TraceId, TraceState, + }, + Context, Key, Value, +}; +use std::convert::TryInto; /// The `ShouldSample` interface allows implementations to provide samplers /// which will return a sampling `SamplingResult` based on information that diff --git a/opentelemetry-sdk/src/trace/tracer.rs b/opentelemetry-sdk/src/trace/tracer.rs index 8860217cd4..c3ccf3a1fa 100644 --- a/opentelemetry-sdk/src/trace/tracer.rs +++ b/opentelemetry-sdk/src/trace/tracer.rs @@ -16,12 +16,12 @@ use crate::{ }, InstrumentationLibrary, }; +use indexmap::IndexMap; use opentelemetry_api::trace::{ Link, SamplingDecision, SamplingResult, SpanBuilder, SpanContext, SpanId, SpanKind, TraceContextExt, TraceFlags, TraceId, TraceState, }; use opentelemetry_api::{Context, Key, KeyValue, Value}; -use indexmap::IndexMap; use std::fmt; use std::sync::Weak; @@ -294,16 +294,19 @@ impl opentelemetry_api::trace::Tracer for Tracer { #[cfg(all(test, feature = "testing", feature = "trace"))] mod tests { - use indexmap::IndexMap; use crate::{ testing::trace::TestSpan, trace::{Config, Sampler, ShouldSample}, InstrumentationLibrary, }; - use opentelemetry_api::{trace::{ - Link, SamplingDecision, SamplingResult, Span, SpanContext, SpanId, SpanKind, - TraceContextExt, TraceFlags, TraceId, TraceState, Tracer, TracerProvider, - }, Context, Value, Key}; + use indexmap::IndexMap; + use opentelemetry_api::{ + trace::{ + Link, SamplingDecision, SamplingResult, Span, SpanContext, SpanId, SpanKind, + TraceContextExt, TraceFlags, TraceId, TraceState, Tracer, TracerProvider, + }, + Context, Key, Value, + }; #[derive(Debug)] struct TestSampler {} From 3d35871cc7e0cd40b9ff83e5cb2698780346497a Mon Sep 17 00:00:00 2001 From: Luca Palmieri Date: Tue, 17 May 2022 18:07:30 +0200 Subject: [PATCH 04/12] Wrap IndexMap to expose only the methods that are insertion-order preserving. --- opentelemetry-api/src/trace/mod.rs | 2 + opentelemetry-api/src/trace/order_map.rs | 625 +++++++++++++++++++++++ opentelemetry-api/src/trace/tracer.rs | 6 +- opentelemetry-sdk/Cargo.toml | 3 +- opentelemetry-sdk/src/trace/sampler.rs | 10 +- opentelemetry-sdk/src/trace/tracer.rs | 10 +- 6 files changed, 640 insertions(+), 16 deletions(-) create mode 100644 opentelemetry-api/src/trace/order_map.rs diff --git a/opentelemetry-api/src/trace/mod.rs b/opentelemetry-api/src/trace/mod.rs index ef377fa5ee..a2825df241 100644 --- a/opentelemetry-api/src/trace/mod.rs +++ b/opentelemetry-api/src/trace/mod.rs @@ -168,6 +168,7 @@ use thiserror::Error; mod context; pub mod noop; +mod order_map; mod span; mod span_context; mod tracer; @@ -175,6 +176,7 @@ mod tracer_provider; pub use self::{ context::{get_active_span, mark_span_as_active, FutureExt, SpanRef, TraceContextExt}, + order_map::OrderMap, span::{Span, SpanKind, Status}, span_context::{SpanContext, SpanId, TraceFlags, TraceId, TraceState}, tracer::{SamplingDecision, SamplingResult, SpanBuilder, Tracer}, diff --git a/opentelemetry-api/src/trace/order_map.rs b/opentelemetry-api/src/trace/order_map.rs new file mode 100644 index 0000000000..adf3c53dfa --- /dev/null +++ b/opentelemetry-api/src/trace/order_map.rs @@ -0,0 +1,625 @@ +use indexmap::map::{ + Drain, Entry, IntoIter, IntoKeys, IntoValues, Iter, IterMut, Keys, Values, ValuesMut, +}; +use indexmap::{Equivalent, IndexMap}; +use std::collections::hash_map::RandomState; +use std::hash::{BuildHasher, Hash}; +use std::iter::FromIterator; +use std::ops::{Index, IndexMut, RangeBounds}; + +/// A hash table implementation that preserves insertion order across all operations. +/// +/// Entries will be returned according to their insertion order when iterating over the collection. +#[derive(Clone, Debug)] +pub struct OrderMap(IndexMap); + +impl OrderMap { + /// Create a new map. (Does not allocate) + #[inline] + pub fn new() -> Self { + Self(IndexMap::new()) + } + + /// Create a new map with capacity for `n` key-value pairs. (Does not + /// allocate if `n` is zero.) + /// + /// Computes in **O(n)** time. + #[inline] + pub fn with_capacity(n: usize) -> Self { + Self(IndexMap::with_capacity(n)) + } +} + +impl OrderMap { + /// Create a new map with capacity for `n` key-value pairs. (Does not + /// allocate if `n` is zero.) + /// + /// Computes in **O(n)** time. + #[inline] + pub fn with_capacity_and_hasher(n: usize, hash_builder: S) -> Self { + Self(IndexMap::with_capacity_and_hasher(n, hash_builder)) + } + + /// Create a new map with `hash_builder`. + /// + /// This function is `const`, so it + /// can be called in `static` contexts. + pub const fn with_hasher(hash_builder: S) -> Self { + Self(IndexMap::with_hasher(hash_builder)) + } + + /// Computes in **O(1)** time. + pub fn capacity(&self) -> usize { + self.0.capacity() + } + + /// Return a reference to the map's `BuildHasher`. + pub fn hasher(&self) -> &S { + &self.0.hasher() + } + + /// Return the number of key-value pairs in the map. + /// + /// Computes in **O(1)** time. + #[inline] + pub fn len(&self) -> usize { + self.0.len() + } + + /// Returns true if the map contains no elements. + /// + /// Computes in **O(1)** time. + #[inline] + pub fn is_empty(&self) -> bool { + self.0.is_empty() + } + + /// Return an iterator over the key-value pairs of the map, in their order + pub fn iter(&self) -> Iter<'_, K, V> { + self.0.iter() + } + + /// Return an iterator over the key-value pairs of the map, in their order + pub fn iter_mut(&mut self) -> IterMut<'_, K, V> { + self.0.iter_mut() + } + + /// Return an iterator over the keys of the map, in their order + pub fn keys(&self) -> Keys<'_, K, V> { + self.0.keys() + } + + /// Return an owning iterator over the keys of the map, in their order + pub fn into_keys(self) -> IntoKeys { + self.0.into_keys() + } + + /// Return an iterator over the values of the map, in their order + pub fn values(&self) -> Values<'_, K, V> { + self.0.values() + } + + /// Return an iterator over mutable references to the values of the map, + /// in their order + pub fn values_mut(&mut self) -> ValuesMut<'_, K, V> { + self.0.values_mut() + } + + /// Return an owning iterator over the values of the map, in their order + pub fn into_values(self) -> IntoValues { + self.0.into_values() + } + + /// Remove all key-value pairs in the map, while preserving its capacity. + /// + /// Computes in **O(n)** time. + pub fn clear(&mut self) { + self.0.clear(); + } + + /// Shortens the map, keeping the first `len` elements and dropping the rest. + /// + /// If `len` is greater than the map's current length, this has no effect. + pub fn truncate(&mut self, len: usize) { + self.0.truncate(len); + } + + /// Clears the `IndexMap` in the given index range, returning those + /// key-value pairs as a drain iterator. + /// + /// The range may be any type that implements `RangeBounds`, + /// including all of the `std::ops::Range*` types, or even a tuple pair of + /// `Bound` start and end values. To drain the map entirely, use `RangeFull` + /// like `map.drain(..)`. + /// + /// This shifts down all entries following the drained range to fill the + /// gap, and keeps the allocated memory for reuse. + /// + /// ***Panics*** if the starting point is greater than the end point or if + /// the end point is greater than the length of the map. + pub fn drain(&mut self, range: R) -> Drain<'_, K, V> + where + R: RangeBounds, + { + self.0.drain(range) + } + + /// Splits the collection into two at the given index. + /// + /// Returns a newly allocated map containing the elements in the range + /// `[at, len)`. After the call, the original map will be left containing + /// the elements `[0, at)` with its previous capacity unchanged. + /// + /// ***Panics*** if `at > len`. + pub fn split_off(&mut self, at: usize) -> Self + where + S: Clone, + { + Self(self.0.split_off(at)) + } +} + +impl OrderMap +where + K: Hash + Eq, + S: BuildHasher, +{ + /// Reserve capacity for `additional` more key-value pairs. + /// + /// Computes in **O(n)** time. + pub fn reserve(&mut self, additional: usize) { + self.0.reserve(additional) + } + + /// Shrink the capacity of the map as much as possible. + /// + /// Computes in **O(n)** time. + pub fn shrink_to_fit(&mut self) { + self.0.shrink_to_fit() + } + + /// Insert a key-value pair in the map. + /// + /// If an equivalent key already exists in the map: the key remains and + /// retains in its place in the order, its corresponding value is updated + /// with `value` and the older value is returned inside `Some(_)`. + /// + /// If no equivalent key existed in the map: the new key-value pair is + /// inserted, last in order, and `None` is returned. + /// + /// Computes in **O(1)** time (amortized average). + /// + /// See also [`entry`](#method.entry) if you you want to insert *or* modify + /// or if you need to get the index of the corresponding key-value pair. + pub fn insert(&mut self, key: K, value: V) -> Option { + self.0.insert(key, value) + } + + /// Insert a key-value pair in the map, and get their index. + /// + /// If an equivalent key already exists in the map: the key remains and + /// retains in its place in the order, its corresponding value is updated + /// with `value` and the older value is returned inside `(index, Some(_))`. + /// + /// If no equivalent key existed in the map: the new key-value pair is + /// inserted, last in order, and `(index, None)` is returned. + /// + /// Computes in **O(1)** time (amortized average). + /// + /// See also [`entry`](#method.entry) if you you want to insert *or* modify + /// or if you need to get the index of the corresponding key-value pair. + pub fn insert_full(&mut self, key: K, value: V) -> (usize, Option) { + self.0.insert_full(key, value) + } + + /// Get the given key’s corresponding entry in the map for insertion and/or + /// in-place manipulation. + /// + /// Computes in **O(1)** time (amortized average). + pub fn entry(&mut self, key: K) -> Entry<'_, K, V> { + self.0.entry(key) + } + + /// Return `true` if an equivalent to `key` exists in the map. + /// + /// Computes in **O(1)** time (average). + pub fn contains_key(&self, key: &Q) -> bool + where + Q: Hash + Equivalent, + { + self.0.contains_key(key) + } + + /// Return a reference to the value stored for `key`, if it is present, + /// else `None`. + /// + /// Computes in **O(1)** time (average). + pub fn get(&self, key: &Q) -> Option<&V> + where + Q: Hash + Equivalent, + { + self.0.get(key) + } + + /// Return references to the key-value pair stored for `key`, + /// if it is present, else `None`. + /// + /// Computes in **O(1)** time (average). + pub fn get_key_value(&self, key: &Q) -> Option<(&K, &V)> + where + Q: Hash + Equivalent, + { + self.0.get_key_value(key) + } + + /// Return item index, key and value + pub fn get_full(&self, key: &Q) -> Option<(usize, &K, &V)> + where + Q: Hash + Equivalent, + { + self.0.get_full(key) + } + + /// Return item index, if it exists in the map + /// + /// Computes in **O(1)** time (average). + pub fn get_index_of(&self, key: &Q) -> Option + where + Q: Hash + Equivalent, + { + self.0.get_index_of(key) + } + + /// Return a mutable reference to the element pointed at by `key`, if it exists. + pub fn get_mut(&mut self, key: &Q) -> Option<&mut V> + where + Q: Hash + Equivalent, + { + self.0.get_mut(key) + } + + /// Return a mutable reference to the element pointed at by `key`, if it exists. + /// It also returns the element's index and its key. + pub fn get_full_mut(&mut self, key: &Q) -> Option<(usize, &K, &mut V)> + where + Q: Hash + Equivalent, + { + self.0.get_full_mut(key) + } + + /// Remove the key-value pair equivalent to `key` and return + /// its value. + /// + /// Like `Vec::remove`, the pair is removed by shifting all of the + /// elements that follow it, preserving their relative order. + /// **This perturbs the index of all of those elements!** + /// + /// Return `None` if `key` is not in map. + /// + /// Computes in **O(n)** time (average). + pub fn shift_remove(&mut self, key: &Q) -> Option + where + Q: Hash + Equivalent, + { + self.0.shift_remove(key) + } + + /// Remove and return the key-value pair equivalent to `key`. + /// + /// Like `Vec::remove`, the pair is removed by shifting all of the + /// elements that follow it, preserving their relative order. + /// **This perturbs the index of all of those elements!** + /// + /// Return `None` if `key` is not in map. + /// + /// Computes in **O(n)** time (average). + pub fn shift_remove_entry(&mut self, key: &Q) -> Option<(K, V)> + where + Q: Hash + Equivalent, + { + self.0.shift_remove_entry(key) + } + + /// Remove the key-value pair equivalent to `key` and return it and + /// the index it had. + /// + /// Like `Vec::remove`, the pair is removed by shifting all of the + /// elements that follow it, preserving their relative order. + /// **This perturbs the index of all of those elements!** + /// + /// Return `None` if `key` is not in map. + /// + /// Computes in **O(n)** time (average). + pub fn shift_remove_full(&mut self, key: &Q) -> Option<(usize, K, V)> + where + Q: Hash + Equivalent, + { + self.0.shift_remove_full(key) + } + + /// Remove the last key-value pair + /// + /// This preserves the order of the remaining elements. + /// + /// Computes in **O(1)** time (average). + pub fn pop(&mut self) -> Option<(K, V)> { + self.0.pop() + } + + /// Scan through each key-value pair in the map and keep those where the + /// closure `keep` returns `true`. + /// + /// The elements are visited in order, and remaining elements keep their + /// order. + /// + /// Computes in **O(n)** time (average). + pub fn retain(&mut self, keep: F) + where + F: FnMut(&K, &mut V) -> bool, + { + self.0.retain(keep); + } +} + +impl OrderMap { + /// Get a key-value pair by index + /// + /// Valid indices are *0 <= index < self.len()* + /// + /// Computes in **O(1)** time. + pub fn get_index(&self, index: usize) -> Option<(&K, &V)> { + self.0.get_index(index) + } + + /// Get a key-value pair by index + /// + /// Valid indices are *0 <= index < self.len()* + /// + /// Computes in **O(1)** time. + pub fn get_index_mut(&mut self, index: usize) -> Option<(&mut K, &mut V)> { + self.0.get_index_mut(index) + } + + /// Get the first key-value pair + /// + /// Computes in **O(1)** time. + pub fn first(&self) -> Option<(&K, &V)> { + self.0.first() + } + + /// Get the first key-value pair, with mutable access to the value + /// + /// Computes in **O(1)** time. + pub fn first_mut(&mut self) -> Option<(&K, &mut V)> { + self.0.first_mut() + } + + /// Get the last key-value pair + /// + /// Computes in **O(1)** time. + pub fn last(&self) -> Option<(&K, &V)> { + self.0.last() + } + + /// Get the last key-value pair, with mutable access to the value + /// + /// Computes in **O(1)** time. + pub fn last_mut(&mut self) -> Option<(&K, &mut V)> { + self.0.last_mut() + } + + /// Remove the key-value pair by index + /// + /// Valid indices are *0 <= index < self.len()* + /// + /// Like `Vec::remove`, the pair is removed by shifting all of the + /// elements that follow it, preserving their relative order. + /// **This perturbs the index of all of those elements!** + /// + /// Computes in **O(n)** time (average). + pub fn shift_remove_index(&mut self, index: usize) -> Option<(K, V)> { + self.0.shift_remove_index(index) + } +} + +impl<'a, K, V, S> IntoIterator for &'a OrderMap { + type Item = (&'a K, &'a V); + type IntoIter = Iter<'a, K, V>; + fn into_iter(self) -> Self::IntoIter { + self.0.iter() + } +} + +impl<'a, K, V, S> IntoIterator for &'a mut OrderMap { + type Item = (&'a K, &'a mut V); + type IntoIter = IterMut<'a, K, V>; + fn into_iter(self) -> Self::IntoIter { + self.0.iter_mut() + } +} + +impl IntoIterator for OrderMap { + type Item = (K, V); + type IntoIter = IntoIter; + fn into_iter(self) -> Self::IntoIter { + self.0.into_iter() + } +} + +/// Access `OrderMap` values corresponding to a key. +/// +/// Panics if the value is missing. +impl Index<&Q> for OrderMap +where + Q: Hash + Equivalent, + K: Hash + Eq, + S: BuildHasher, +{ + type Output = V; + + /// Returns a reference to the value corresponding to the supplied `key`. + /// + /// ***Panics*** if `key` is not present in the map. + fn index(&self, key: &Q) -> &V { + self.0.index(key) + } +} + +/// Access `Ordermap` values corresponding to a key. +/// +/// Mutable indexing allows changing / updating values of key-value +/// pairs that are already present. +/// +/// You can **not** insert new pairs with index syntax, use `.insert()`. +impl IndexMut<&Q> for OrderMap +where + Q: Hash + Equivalent, + K: Hash + Eq, + S: BuildHasher, +{ + /// Returns a mutable reference to the value corresponding to the supplied `key`. + /// + /// ***Panics*** if `key` is not present in the map. + fn index_mut(&mut self, key: &Q) -> &mut V { + self.0.index_mut(key) + } +} + +/// Access `IndexMap` values at indexed positions. +/// +/// It panics if the index is out of bounds. +impl Index for OrderMap { + type Output = V; + + /// Returns a reference to the value at the supplied `index`. + /// + /// ***Panics*** if `index` is out of bounds. + fn index(&self, index: usize) -> &V { + self.0.index(index) + } +} + +/// Access `IndexMap` values at indexed positions. +/// +/// Mutable indexing allows changing / updating indexed values +/// that are already present. +/// +/// You can **not** insert new values with index syntax, use `.insert()`. +/// +/// # Examples +/// +/// ``` +/// use indexmap::IndexMap; +/// +/// let mut map = IndexMap::new(); +/// for word in "Lorem ipsum dolor sit amet".split_whitespace() { +/// map.insert(word.to_lowercase(), word.to_string()); +/// } +/// let lorem = &mut map[0]; +/// assert_eq!(lorem, "Lorem"); +/// lorem.retain(char::is_lowercase); +/// assert_eq!(map["lorem"], "orem"); +/// ``` +/// +/// ```should_panic +/// use indexmap::IndexMap; +/// +/// let mut map = IndexMap::new(); +/// map.insert("foo", 1); +/// map[10] = 1; // panics! +/// ``` +impl IndexMut for OrderMap { + /// Returns a mutable reference to the value at the supplied `index`. + /// + /// ***Panics*** if `index` is out of bounds. + fn index_mut(&mut self, index: usize) -> &mut V { + self.0.index_mut(index) + } +} + +impl FromIterator<(K, V)> for OrderMap +where + K: Hash + Eq, + S: BuildHasher + Default, +{ + /// Create an `OrderMap` from the sequence of key-value pairs in the + /// iterable. + /// + /// `from_iter` uses the same logic as `extend`. See + /// [`extend`](#method.extend) for more details. + fn from_iter>(iterable: I) -> Self { + Self(IndexMap::from_iter(iterable)) + } +} + +impl From<[(K, V); N]> for OrderMap +where + K: Hash + Eq, +{ + fn from(arr: [(K, V); N]) -> Self { + Self(IndexMap::from(arr)) + } +} + +impl Extend<(K, V)> for OrderMap +where + K: Hash + Eq, + S: BuildHasher, +{ + /// Extend the map with all key-value pairs in the iterable. + /// + /// This is equivalent to calling [`insert`](#method.insert) for each of + /// them in order, which means that for keys that already existed + /// in the map, their value is updated but it keeps the existing order. + /// + /// New keys are inserted in the order they appear in the sequence. If + /// equivalents of a key occur more than once, the last corresponding value + /// prevails. + fn extend>(&mut self, iterable: I) { + self.0.extend(iterable) + } +} + +impl<'a, K, V, S> Extend<(&'a K, &'a V)> for OrderMap +where + K: 'a + Hash + Eq + Copy, + V: 'a + Copy, + S: BuildHasher, +{ + /// Extend the map with all key-value pairs in the iterable. + /// + /// See the first extend method for more details. + fn extend>(&mut self, iterable: I) { + self.0.extend(iterable) + } +} + +impl Default for OrderMap +where + S: Default, +{ + /// Return an empty `OrderMap` + fn default() -> Self { + Self(IndexMap::default()) + } +} + +impl PartialEq> for OrderMap +where + K: Hash + Eq, + V1: PartialEq, + S1: BuildHasher, + S2: BuildHasher, +{ + fn eq(&self, other: &OrderMap) -> bool { + self.0.eq(&other.0) + } +} + +impl Eq for OrderMap +where + K: Eq + Hash, + V: Eq, + S: BuildHasher, +{ +} diff --git a/opentelemetry-api/src/trace/tracer.rs b/opentelemetry-api/src/trace/tracer.rs index c90d3011ee..796018a4de 100644 --- a/opentelemetry-api/src/trace/tracer.rs +++ b/opentelemetry-api/src/trace/tracer.rs @@ -1,8 +1,8 @@ +use crate::trace::OrderMap; use crate::{ trace::{Event, Link, Span, SpanId, SpanKind, Status, TraceContextExt, TraceId, TraceState}, Context, Key, KeyValue, Value, }; -use indexmap::IndexMap; use std::borrow::Cow; use std::time::SystemTime; @@ -260,7 +260,7 @@ pub struct SpanBuilder { pub end_time: Option, /// Span attributes - pub attributes: Option>, + pub attributes: Option>, /// Span events pub events: Option>, @@ -326,7 +326,7 @@ impl SpanBuilder { } /// Assign span attributes - pub fn with_attributes(self, attributes: IndexMap) -> Self { + pub fn with_attributes(self, attributes: OrderMap) -> Self { SpanBuilder { attributes: Some(attributes), ..self diff --git a/opentelemetry-sdk/Cargo.toml b/opentelemetry-sdk/Cargo.toml index 090c9b23c3..7d810cbdfc 100644 --- a/opentelemetry-sdk/Cargo.toml +++ b/opentelemetry-sdk/Cargo.toml @@ -12,7 +12,6 @@ fnv = { version = "1.0", optional = true } futures-channel = "0.3" futures-executor = "0.3" futures-util = { version = "0.3", default-features = false, features = ["std", "sink", "async-await-macro"] } -indexmap = { version = "1", optional = true } lazy_static = "1.4" once_cell = "1.10" opentelemetry-api = { version = "0.1", path = "../opentelemetry-api/" } @@ -35,7 +34,7 @@ rand_distr = "0.4.0" [features] default = ["trace"] -trace = ["opentelemetry-api/trace", "crossbeam-channel", "rand", "pin-project", "async-trait", "percent-encoding", "indexmap"] +trace = ["opentelemetry-api/trace", "crossbeam-channel", "rand", "pin-project", "async-trait", "percent-encoding"] metrics = ["opentelemetry-api/metrics", "dashmap", "fnv"] testing = ["opentelemetry-api/testing", "trace", "metrics", "rt-async-std", "rt-tokio", "rt-tokio-current-thread", "tokio/macros", "tokio/rt-multi-thread"] rt-tokio = ["tokio", "tokio-stream"] diff --git a/opentelemetry-sdk/src/trace/sampler.rs b/opentelemetry-sdk/src/trace/sampler.rs index 65efd03a63..6d3e4942d6 100644 --- a/opentelemetry-sdk/src/trace/sampler.rs +++ b/opentelemetry-sdk/src/trace/sampler.rs @@ -38,7 +38,7 @@ //! MUST NOT allow this combination. use crate::InstrumentationLibrary; -use indexmap::IndexMap; +use opentelemetry_api::trace::OrderMap; use opentelemetry_api::{ trace::{ Link, SamplingDecision, SamplingResult, SpanKind, TraceContextExt, TraceId, TraceState, @@ -59,7 +59,7 @@ pub trait ShouldSample: Send + Sync + std::fmt::Debug { trace_id: TraceId, name: &str, span_kind: &SpanKind, - attributes: &IndexMap, + attributes: &OrderMap, links: &[Link], instrumentation_library: &InstrumentationLibrary, ) -> SamplingResult; @@ -87,7 +87,7 @@ impl ShouldSample for Sampler { trace_id: TraceId, name: &str, span_kind: &SpanKind, - attributes: &IndexMap, + attributes: &OrderMap, links: &[Link], instrumentation_library: &InstrumentationLibrary, ) -> SamplingResult { @@ -243,7 +243,7 @@ mod tests { trace_id, name, &SpanKind::Internal, - &IndexMap::new(), + &Default::default(), &[], &InstrumentationLibrary::default(), ) @@ -285,7 +285,7 @@ mod tests { TraceId::from_u128(1), "should sample", &SpanKind::Internal, - &IndexMap::new(), + &Default::default(), &[], &instrumentation_library, ); diff --git a/opentelemetry-sdk/src/trace/tracer.rs b/opentelemetry-sdk/src/trace/tracer.rs index c3ccf3a1fa..2852d621d7 100644 --- a/opentelemetry-sdk/src/trace/tracer.rs +++ b/opentelemetry-sdk/src/trace/tracer.rs @@ -16,9 +16,8 @@ use crate::{ }, InstrumentationLibrary, }; -use indexmap::IndexMap; use opentelemetry_api::trace::{ - Link, SamplingDecision, SamplingResult, SpanBuilder, SpanContext, SpanId, SpanKind, + Link, OrderMap, SamplingDecision, SamplingResult, SpanBuilder, SpanContext, SpanId, SpanKind, TraceContextExt, TraceFlags, TraceId, TraceState, }; use opentelemetry_api::{Context, Key, KeyValue, Value}; @@ -73,7 +72,7 @@ impl Tracer { trace_id: TraceId, name: &str, span_kind: &SpanKind, - attributes: &IndexMap, + attributes: &OrderMap, links: &[Link], config: &Config, instrumentation_library: &InstrumentationLibrary, @@ -299,10 +298,9 @@ mod tests { trace::{Config, Sampler, ShouldSample}, InstrumentationLibrary, }; - use indexmap::IndexMap; use opentelemetry_api::{ trace::{ - Link, SamplingDecision, SamplingResult, Span, SpanContext, SpanId, SpanKind, + Link, OrderMap, SamplingDecision, SamplingResult, Span, SpanContext, SpanId, SpanKind, TraceContextExt, TraceFlags, TraceId, TraceState, Tracer, TracerProvider, }, Context, Key, Value, @@ -318,7 +316,7 @@ mod tests { _trace_id: TraceId, _name: &str, _span_kind: &SpanKind, - _attributes: &IndexMap, + _attributes: &OrderMap, _links: &[Link], _instrumentation_library: &InstrumentationLibrary, ) -> SamplingResult { From 90516dc965211339c5ef10fde7b09c4010ec6e21 Mon Sep 17 00:00:00 2001 From: Luca Palmieri Date: Tue, 17 May 2022 18:11:08 +0200 Subject: [PATCH 05/12] Raise MSRV to 1.51 to get support for const generics. --- .github/workflows/ci.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 5b5d791054..6cfd02233e 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -68,7 +68,7 @@ jobs: - uses: actions-rs/toolchain@v1 with: profile: minimal - toolchain: 1.49.0 + toolchain: 1.51.0 override: true - name: Run tests run: cargo --version && From 19831d457b268de8b31dcdf5ae37208d49dffee1 Mon Sep 17 00:00:00 2001 From: Luca Palmieri Date: Tue, 17 May 2022 18:30:20 +0200 Subject: [PATCH 06/12] Fix doctest. --- opentelemetry-semantic-conventions/src/trace.rs | 12 +++++++----- 1 file changed, 7 insertions(+), 5 deletions(-) diff --git a/opentelemetry-semantic-conventions/src/trace.rs b/opentelemetry-semantic-conventions/src/trace.rs index f6371d0763..b4188f908c 100644 --- a/opentelemetry-semantic-conventions/src/trace.rs +++ b/opentelemetry-semantic-conventions/src/trace.rs @@ -14,16 +14,18 @@ //! ## Usage //! //! ```rust -//! use opentelemetry::{global, trace::Tracer as _}; +//! use opentelemetry::{global, trace::Tracer as _, trace::OrderMap}; //! use opentelemetry_semantic_conventions as semcov; //! //! let tracer = global::tracer("my-component"); +//! let net_peer_ip = semcov::trace::NET_PEER_IP.string("10.0.0.1"); +//! let net_port = semcov::trace::NET_PEER_PORT.i64(80); //! let _span = tracer //! .span_builder("span-name") -//! .with_attributes(vec![ -//! semcov::trace::NET_PEER_IP.string("10.0.0.1"), -//! semcov::trace::NET_PEER_PORT.i64(80), -//! ]) +//! .with_attributes(OrderMap::from([ +//! (net_peer_ip.key, net_peer_ip.value), +//! (net_port.key, net_port.value), +//! ])) //! .start(&tracer); //! ``` From 0e897a7bb863973c1554bef0f2d695ba547440ce Mon Sep 17 00:00:00 2001 From: Luca Palmieri Date: Tue, 17 May 2022 18:30:56 +0200 Subject: [PATCH 07/12] Fix lint. --- opentelemetry-api/src/trace/order_map.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/opentelemetry-api/src/trace/order_map.rs b/opentelemetry-api/src/trace/order_map.rs index adf3c53dfa..2a5aa5eec4 100644 --- a/opentelemetry-api/src/trace/order_map.rs +++ b/opentelemetry-api/src/trace/order_map.rs @@ -55,7 +55,7 @@ impl OrderMap { /// Return a reference to the map's `BuildHasher`. pub fn hasher(&self) -> &S { - &self.0.hasher() + self.0.hasher() } /// Return the number of key-value pairs in the map. From c348673f904f8587eb865a4a9e85fe1db26a0f2e Mon Sep 17 00:00:00 2001 From: Luca Palmieri Date: Thu, 19 May 2022 15:49:59 +0200 Subject: [PATCH 08/12] Add specialised implementations to make it easier to work with KeyValue iterators/collections. --- opentelemetry-api/src/trace/order_map.rs | 43 +++++++++++++++++++ .../src/trace.rs | 6 +-- 2 files changed, 45 insertions(+), 4 deletions(-) diff --git a/opentelemetry-api/src/trace/order_map.rs b/opentelemetry-api/src/trace/order_map.rs index 2a5aa5eec4..bd24e8f941 100644 --- a/opentelemetry-api/src/trace/order_map.rs +++ b/opentelemetry-api/src/trace/order_map.rs @@ -1,3 +1,4 @@ +use crate::{Key, KeyValue, Value}; use indexmap::map::{ Drain, Entry, IntoIter, IntoKeys, IntoValues, Iter, IterMut, Keys, Values, ValuesMut, }; @@ -623,3 +624,45 @@ where S: BuildHasher, { } + +impl FromIterator for OrderMap +where + S: BuildHasher + Default, +{ + /// Create an `OrderMap` from the sequence of key-value pairs in the + /// iterable. + /// + /// `from_iter` uses the same logic as `extend`. See + /// [`extend`](#method.extend) for more details. + fn from_iter>(iterable: I) -> Self { + Self(IndexMap::from_iter( + iterable.into_iter().map(|kv| (kv.key, kv.value)), + )) + } +} + +impl From<[KeyValue; N]> for OrderMap { + fn from(arr: [KeyValue; N]) -> Self { + let arr = arr.map(|kv| (kv.key, kv.value)); + Self(IndexMap::from(arr)) + } +} + +impl Extend for OrderMap +where + S: BuildHasher, +{ + /// Extend the map with all key-value pairs in the iterable. + /// + /// This is equivalent to calling [`insert`](#method.insert) for each of + /// them in order, which means that for keys that already existed + /// in the map, their value is updated but it keeps the existing order. + /// + /// New keys are inserted in the order they appear in the sequence. If + /// equivalents of a key occur more than once, the last corresponding value + /// prevails. + fn extend>(&mut self, iterable: I) { + self.0 + .extend(iterable.into_iter().map(|kv| (kv.key, kv.value))) + } +} diff --git a/opentelemetry-semantic-conventions/src/trace.rs b/opentelemetry-semantic-conventions/src/trace.rs index b4188f908c..c14671c3ac 100644 --- a/opentelemetry-semantic-conventions/src/trace.rs +++ b/opentelemetry-semantic-conventions/src/trace.rs @@ -18,13 +18,11 @@ //! use opentelemetry_semantic_conventions as semcov; //! //! let tracer = global::tracer("my-component"); -//! let net_peer_ip = semcov::trace::NET_PEER_IP.string("10.0.0.1"); -//! let net_port = semcov::trace::NET_PEER_PORT.i64(80); //! let _span = tracer //! .span_builder("span-name") //! .with_attributes(OrderMap::from([ -//! (net_peer_ip.key, net_peer_ip.value), -//! (net_port.key, net_port.value), +//! semcov::trace::NET_PEER_IP.string("10.0.0.1"), +//! semcov::trace::NET_PEER_PORT.i64(80), //! ])) //! .start(&tracer); //! ``` From d08c9bbf3ba8f9acf614ba1ae3e7eaacef6271aa Mon Sep 17 00:00:00 2001 From: Luca Palmieri Date: Thu, 19 May 2022 15:56:15 +0200 Subject: [PATCH 09/12] Bump MSRV to get access to array::map. --- .github/workflows/ci.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 6cfd02233e..4779b29d8d 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -68,7 +68,7 @@ jobs: - uses: actions-rs/toolchain@v1 with: profile: minimal - toolchain: 1.51.0 + toolchain: 1.55.0 override: true - name: Run tests run: cargo --version && From 27226209456068411259a3789622790f10e54268 Mon Sep 17 00:00:00 2001 From: Luca Palmieri Date: Fri, 20 May 2022 09:29:07 +0200 Subject: [PATCH 10/12] Minimise breakages for existing users. --- opentelemetry-api/src/trace/tracer.rs | 22 ++++++++++++++++++++-- 1 file changed, 20 insertions(+), 2 deletions(-) diff --git a/opentelemetry-api/src/trace/tracer.rs b/opentelemetry-api/src/trace/tracer.rs index 796018a4de..66b03ab2cf 100644 --- a/opentelemetry-api/src/trace/tracer.rs +++ b/opentelemetry-api/src/trace/tracer.rs @@ -4,6 +4,7 @@ use crate::{ Context, Key, KeyValue, Value, }; use std::borrow::Cow; +use std::iter::FromIterator; use std::time::SystemTime; /// The interface for constructing [`Span`]s. @@ -325,8 +326,25 @@ impl SpanBuilder { } } - /// Assign span attributes - pub fn with_attributes(self, attributes: OrderMap) -> Self { + /// Assign span attributes from an iterable. + /// + /// Check out [`SpanBuilder::with_attributes_map`] to assign span attributes + /// via an [`OrderMap`] instance. + pub fn with_attributes(self, attributes: I) -> Self + where + I: IntoIterator + { + SpanBuilder { + attributes: Some(OrderMap::from_iter(attributes.into_iter())), + ..self + } + } + + /// Assign span attributes. + /// + /// Check out [`SpanBuilder::with_attributes`] to assign span attributes + /// from an iterable of [`KeyValue`]s. + pub fn with_attributes_map(self, attributes: OrderMap) -> Self { SpanBuilder { attributes: Some(attributes), ..self From 3a6d7a3aca9337119681b3bf0263289469131342 Mon Sep 17 00:00:00 2001 From: Luca Palmieri Date: Fri, 20 May 2022 10:08:09 +0200 Subject: [PATCH 11/12] Fix invocation. --- opentelemetry-semantic-conventions/src/trace.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/opentelemetry-semantic-conventions/src/trace.rs b/opentelemetry-semantic-conventions/src/trace.rs index c14671c3ac..01448e16d8 100644 --- a/opentelemetry-semantic-conventions/src/trace.rs +++ b/opentelemetry-semantic-conventions/src/trace.rs @@ -20,10 +20,10 @@ //! let tracer = global::tracer("my-component"); //! let _span = tracer //! .span_builder("span-name") -//! .with_attributes(OrderMap::from([ +//! .with_attributes([ //! semcov::trace::NET_PEER_IP.string("10.0.0.1"), //! semcov::trace::NET_PEER_PORT.i64(80), -//! ])) +//! ]) //! .start(&tracer); //! ``` From e39fdd4a994b0e5c30344e344388c72174188b0b Mon Sep 17 00:00:00 2001 From: Luca Palmieri Date: Fri, 20 May 2022 10:13:56 +0200 Subject: [PATCH 12/12] Rustfmt --- opentelemetry-api/src/trace/tracer.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/opentelemetry-api/src/trace/tracer.rs b/opentelemetry-api/src/trace/tracer.rs index 66b03ab2cf..e36d32c5c5 100644 --- a/opentelemetry-api/src/trace/tracer.rs +++ b/opentelemetry-api/src/trace/tracer.rs @@ -332,7 +332,7 @@ impl SpanBuilder { /// via an [`OrderMap`] instance. pub fn with_attributes(self, attributes: I) -> Self where - I: IntoIterator + I: IntoIterator, { SpanBuilder { attributes: Some(OrderMap::from_iter(attributes.into_iter())),