Skip to content

Commit febfe76

Browse files
authored
Auto merge of #37326 - SimonSapin:from-cow, r=alexcrichton
Implement `From<Cow<str>> for String` and `From<Cow<[T]>> for Vec<T>`. Motivation: the `selectors` crate is generic over a string type, in order to support all of `String`, `string_cache::Atom`, and `gecko_string_cache::Atom`. Multiple trait bounds are used for the various operations done with these strings. One of these operations is creating a string (as efficiently as possible, re-using an existing memory allocation if possible) from `Cow<str>`. The `std::convert::From` trait seems natural for this, but the relevant implementation was missing before this PR. To work around this I’ve added a `FromCowStr` trait in `selectors`, but with trait coherence that means one of `selectors` or `string_cache` needs to depend on the other to implement this trait. Using a trait from `std` would solve this. The `Vec<T>` implementation is just added for consistency. I also tried a more general `impl<'a, O, B: ?Sized + ToOwned<Owned=O>> From<Cow<'a, B>> for O`, but (the compiler thinks?) it conflicts with `From<T> for T` the impl (after moving all of `collections::borrow` into `core::borrow` to work around trait coherence).
2 parents 4845add + 7e603d4 commit febfe76

File tree

4 files changed

+28
-0
lines changed

4 files changed

+28
-0
lines changed

src/libcollections/string.rs

+7
Original file line numberDiff line numberDiff line change
@@ -1860,6 +1860,13 @@ impl<'a> From<&'a str> for String {
18601860
}
18611861
}
18621862

1863+
#[stable(feature = "string_from_cow_str", since = "1.14.0")]
1864+
impl<'a> From<Cow<'a, str>> for String {
1865+
fn from(s: Cow<'a, str>) -> String {
1866+
s.into_owned()
1867+
}
1868+
}
1869+
18631870
#[stable(feature = "rust1", since = "1.0.0")]
18641871
impl<'a> From<&'a str> for Cow<'a, str> {
18651872
#[inline]

src/libcollections/vec.rs

+7
Original file line numberDiff line numberDiff line change
@@ -1797,6 +1797,13 @@ impl<'a, T: Clone> From<&'a [T]> for Vec<T> {
17971797
}
17981798
}
17991799

1800+
#[stable(feature = "vec_from_cow_slice", since = "1.14.0")]
1801+
impl<'a, T> From<Cow<'a, [T]>> for Vec<T> where [T]: ToOwned<Owned=Vec<T>> {
1802+
fn from(s: Cow<'a, [T]>) -> Vec<T> {
1803+
s.into_owned()
1804+
}
1805+
}
1806+
18001807
#[stable(feature = "rust1", since = "1.0.0")]
18011808
impl<'a> From<&'a str> for Vec<u8> {
18021809
fn from(s: &'a str) -> Vec<u8> {

src/libcollectionstest/string.rs

+6
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,12 @@ fn test_from_str() {
3535
assert_eq!(owned.as_ref().map(|s| &**s), Some("string"));
3636
}
3737

38+
#[test]
39+
fn test_from_cow_str() {
40+
assert_eq!(String::from(Cow::Borrowed("string")), "string");
41+
assert_eq!(String::from(Cow::Owned(String::from("string"))), "string");
42+
}
43+
3844
#[test]
3945
fn test_unsized_to_string() {
4046
let s: &str = "abc";

src/libcollectionstest/vec.rs

+8
Original file line numberDiff line numberDiff line change
@@ -597,6 +597,14 @@ fn test_cow_from() {
597597
}
598598
}
599599

600+
#[test]
601+
fn test_from_cow() {
602+
let borrowed: &[_] = &["borrowed", "(slice)"];
603+
let owned = vec!["owned", "(vec)"];
604+
assert_eq!(Vec::from(Cow::Borrowed(borrowed)), vec!["borrowed", "(slice)"]);
605+
assert_eq!(Vec::from(Cow::Owned(owned)), vec!["owned", "(vec)"]);
606+
}
607+
600608
#[allow(dead_code)]
601609
fn assert_covariance() {
602610
fn drain<'new>(d: Drain<'static, &'static str>) -> Drain<'new, &'new str> { d }

0 commit comments

Comments
 (0)