2014-04-02 16:54:22 -07:00
|
|
|
|
// Copyright 2014 The Rust Project Developers. See the COPYRIGHT
|
|
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
|
//
|
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
|
// except according to those terms.
|
|
|
|
|
|
|
|
|
|
//! An owned, growable string that enforces that its contents are valid UTF-8.
|
|
|
|
|
|
std: Recreate a `collections` module
As with the previous commit with `librand`, this commit shuffles around some
`collections` code. The new state of the world is similar to that of librand:
* The libcollections crate now only depends on libcore and liballoc.
* The standard library has a new module, `std::collections`. All functionality
of libcollections is reexported through this module.
I would like to stress that this change is purely cosmetic. There are very few
alterations to these primitives.
There are a number of notable points about the new organization:
* std::{str, slice, string, vec} all moved to libcollections. There is no reason
that these primitives shouldn't be necessarily usable in a freestanding
context that has allocation. These are all reexported in their usual places in
the standard library.
* The `hashmap`, and transitively the `lru_cache`, modules no longer reside in
`libcollections`, but rather in libstd. The reason for this is because the
`HashMap::new` contructor requires access to the OSRng for initially seeding
the hash map. Beyond this requirement, there is no reason that the hashmap
could not move to libcollections.
I do, however, have a plan to move the hash map to the collections module. The
`HashMap::new` function could be altered to require that the `H` hasher
parameter ascribe to the `Default` trait, allowing the entire `hashmap` module
to live in libcollections. The key idea would be that the default hasher would
be different in libstd. Something along the lines of:
// src/libstd/collections/mod.rs
pub type HashMap<K, V, H = RandomizedSipHasher> =
core_collections::HashMap<K, V, H>;
This is not possible today because you cannot invoke static methods through
type aliases. If we modified the compiler, however, to allow invocation of
static methods through type aliases, then this type definition would
essentially be switching the default hasher from `SipHasher` in libcollections
to a libstd-defined `RandomizedSipHasher` type. This type's `Default`
implementation would randomly seed the `SipHasher` instance, and otherwise
perform the same as `SipHasher`.
This future state doesn't seem incredibly far off, but until that time comes,
the hashmap module will live in libstd to not compromise on functionality.
* In preparation for the hashmap moving to libcollections, the `hash` module has
moved from libstd to libcollections. A previously snapshotted commit enables a
distinct `Writer` trait to live in the `hash` module which `Hash`
implementations are now parameterized over.
Due to using a custom trait, the `SipHasher` implementation has lost its
specialized methods for writing integers. These can be re-added
backwards-compatibly in the future via default methods if necessary, but the
FNV hashing should satisfy much of the need for speedier hashing.
A list of breaking changes:
* HashMap::{get, get_mut} no longer fails with the key formatted into the error
message with `{:?}`, instead, a generic message is printed. With backtraces,
it should still be not-too-hard to track down errors.
* The HashMap, HashSet, and LruCache types are now available through
std::collections instead of the collections crate.
* Manual implementations of hash should be parameterized over `hash::Writer`
instead of just `Writer`.
[breaking-change]
2014-05-29 18:50:12 -07:00
|
|
|
|
use core::prelude::*;
|
|
|
|
|
|
|
|
|
|
use core::default::Default;
|
|
|
|
|
use core::fmt;
|
|
|
|
|
use core::mem;
|
|
|
|
|
use core::ptr;
|
|
|
|
|
use core::raw::Slice;
|
|
|
|
|
|
2014-06-06 16:33:44 -07:00
|
|
|
|
use {Collection, Mutable};
|
std: Recreate a `collections` module
As with the previous commit with `librand`, this commit shuffles around some
`collections` code. The new state of the world is similar to that of librand:
* The libcollections crate now only depends on libcore and liballoc.
* The standard library has a new module, `std::collections`. All functionality
of libcollections is reexported through this module.
I would like to stress that this change is purely cosmetic. There are very few
alterations to these primitives.
There are a number of notable points about the new organization:
* std::{str, slice, string, vec} all moved to libcollections. There is no reason
that these primitives shouldn't be necessarily usable in a freestanding
context that has allocation. These are all reexported in their usual places in
the standard library.
* The `hashmap`, and transitively the `lru_cache`, modules no longer reside in
`libcollections`, but rather in libstd. The reason for this is because the
`HashMap::new` contructor requires access to the OSRng for initially seeding
the hash map. Beyond this requirement, there is no reason that the hashmap
could not move to libcollections.
I do, however, have a plan to move the hash map to the collections module. The
`HashMap::new` function could be altered to require that the `H` hasher
parameter ascribe to the `Default` trait, allowing the entire `hashmap` module
to live in libcollections. The key idea would be that the default hasher would
be different in libstd. Something along the lines of:
// src/libstd/collections/mod.rs
pub type HashMap<K, V, H = RandomizedSipHasher> =
core_collections::HashMap<K, V, H>;
This is not possible today because you cannot invoke static methods through
type aliases. If we modified the compiler, however, to allow invocation of
static methods through type aliases, then this type definition would
essentially be switching the default hasher from `SipHasher` in libcollections
to a libstd-defined `RandomizedSipHasher` type. This type's `Default`
implementation would randomly seed the `SipHasher` instance, and otherwise
perform the same as `SipHasher`.
This future state doesn't seem incredibly far off, but until that time comes,
the hashmap module will live in libstd to not compromise on functionality.
* In preparation for the hashmap moving to libcollections, the `hash` module has
moved from libstd to libcollections. A previously snapshotted commit enables a
distinct `Writer` trait to live in the `hash` module which `Hash`
implementations are now parameterized over.
Due to using a custom trait, the `SipHasher` implementation has lost its
specialized methods for writing integers. These can be re-added
backwards-compatibly in the future via default methods if necessary, but the
FNV hashing should satisfy much of the need for speedier hashing.
A list of breaking changes:
* HashMap::{get, get_mut} no longer fails with the key formatted into the error
message with `{:?}`, instead, a generic message is printed. With backtraces,
it should still be not-too-hard to track down errors.
* The HashMap, HashSet, and LruCache types are now available through
std::collections instead of the collections crate.
* Manual implementations of hash should be parameterized over `hash::Writer`
instead of just `Writer`.
[breaking-change]
2014-05-29 18:50:12 -07:00
|
|
|
|
use hash;
|
2014-04-10 20:55:34 +10:00
|
|
|
|
use str;
|
std: Recreate a `collections` module
As with the previous commit with `librand`, this commit shuffles around some
`collections` code. The new state of the world is similar to that of librand:
* The libcollections crate now only depends on libcore and liballoc.
* The standard library has a new module, `std::collections`. All functionality
of libcollections is reexported through this module.
I would like to stress that this change is purely cosmetic. There are very few
alterations to these primitives.
There are a number of notable points about the new organization:
* std::{str, slice, string, vec} all moved to libcollections. There is no reason
that these primitives shouldn't be necessarily usable in a freestanding
context that has allocation. These are all reexported in their usual places in
the standard library.
* The `hashmap`, and transitively the `lru_cache`, modules no longer reside in
`libcollections`, but rather in libstd. The reason for this is because the
`HashMap::new` contructor requires access to the OSRng for initially seeding
the hash map. Beyond this requirement, there is no reason that the hashmap
could not move to libcollections.
I do, however, have a plan to move the hash map to the collections module. The
`HashMap::new` function could be altered to require that the `H` hasher
parameter ascribe to the `Default` trait, allowing the entire `hashmap` module
to live in libcollections. The key idea would be that the default hasher would
be different in libstd. Something along the lines of:
// src/libstd/collections/mod.rs
pub type HashMap<K, V, H = RandomizedSipHasher> =
core_collections::HashMap<K, V, H>;
This is not possible today because you cannot invoke static methods through
type aliases. If we modified the compiler, however, to allow invocation of
static methods through type aliases, then this type definition would
essentially be switching the default hasher from `SipHasher` in libcollections
to a libstd-defined `RandomizedSipHasher` type. This type's `Default`
implementation would randomly seed the `SipHasher` instance, and otherwise
perform the same as `SipHasher`.
This future state doesn't seem incredibly far off, but until that time comes,
the hashmap module will live in libstd to not compromise on functionality.
* In preparation for the hashmap moving to libcollections, the `hash` module has
moved from libstd to libcollections. A previously snapshotted commit enables a
distinct `Writer` trait to live in the `hash` module which `Hash`
implementations are now parameterized over.
Due to using a custom trait, the `SipHasher` implementation has lost its
specialized methods for writing integers. These can be re-added
backwards-compatibly in the future via default methods if necessary, but the
FNV hashing should satisfy much of the need for speedier hashing.
A list of breaking changes:
* HashMap::{get, get_mut} no longer fails with the key formatted into the error
message with `{:?}`, instead, a generic message is printed. With backtraces,
it should still be not-too-hard to track down errors.
* The HashMap, HashSet, and LruCache types are now available through
std::collections instead of the collections crate.
* Manual implementations of hash should be parameterized over `hash::Writer`
instead of just `Writer`.
[breaking-change]
2014-05-29 18:50:12 -07:00
|
|
|
|
use str::{CharRange, StrAllocating};
|
2014-04-02 16:54:22 -07:00
|
|
|
|
use vec::Vec;
|
|
|
|
|
|
2014-04-10 20:55:34 +10:00
|
|
|
|
/// A growable string stored as a UTF-8 encoded buffer.
|
2014-05-31 10:43:52 -07:00
|
|
|
|
#[deriving(Clone, PartialEq, PartialOrd, Eq, Ord)]
|
2014-05-22 16:57:53 -07:00
|
|
|
|
pub struct String {
|
2014-04-02 16:54:22 -07:00
|
|
|
|
vec: Vec<u8>,
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-22 16:57:53 -07:00
|
|
|
|
impl String {
|
2014-04-21 00:49:39 -04:00
|
|
|
|
/// Creates a new string buffer initialized with the empty string.
|
2014-04-02 16:54:22 -07:00
|
|
|
|
#[inline]
|
2014-05-22 16:57:53 -07:00
|
|
|
|
pub fn new() -> String {
|
|
|
|
|
String {
|
2014-04-02 16:54:22 -07:00
|
|
|
|
vec: Vec::new(),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Creates a new string buffer with the given capacity.
|
|
|
|
|
#[inline]
|
2014-05-22 16:57:53 -07:00
|
|
|
|
pub fn with_capacity(capacity: uint) -> String {
|
|
|
|
|
String {
|
2014-04-02 16:54:22 -07:00
|
|
|
|
vec: Vec::with_capacity(capacity),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Creates a new string buffer from length, capacity, and a pointer.
|
|
|
|
|
#[inline]
|
2014-05-22 16:57:53 -07:00
|
|
|
|
pub unsafe fn from_raw_parts(length: uint, capacity: uint, ptr: *mut u8) -> String {
|
|
|
|
|
String {
|
2014-04-02 16:54:22 -07:00
|
|
|
|
vec: Vec::from_raw_parts(length, capacity, ptr),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Creates a new string buffer from the given string.
|
|
|
|
|
#[inline]
|
2014-05-22 16:57:53 -07:00
|
|
|
|
pub fn from_str(string: &str) -> String {
|
|
|
|
|
String {
|
2014-04-02 16:54:22 -07:00
|
|
|
|
vec: Vec::from_slice(string.as_bytes())
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-24 21:59:56 -07:00
|
|
|
|
#[allow(missing_doc)]
|
|
|
|
|
#[deprecated = "obsoleted by the removal of ~str"]
|
2014-04-02 16:54:22 -07:00
|
|
|
|
#[inline]
|
2014-05-22 16:57:53 -07:00
|
|
|
|
pub fn from_owned_str(string: String) -> String {
|
2014-05-19 17:23:26 -07:00
|
|
|
|
string
|
2014-04-02 16:54:22 -07:00
|
|
|
|
}
|
|
|
|
|
|
2014-05-14 16:55:24 -07:00
|
|
|
|
/// Returns the vector as a string buffer, if possible, taking care not to
|
|
|
|
|
/// copy it.
|
|
|
|
|
///
|
|
|
|
|
/// Returns `Err` with the original vector if the vector contains invalid
|
|
|
|
|
/// UTF-8.
|
2014-06-30 16:41:30 +02:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```rust
|
|
|
|
|
/// let hello_vec = vec![104, 101, 108, 108, 111];
|
|
|
|
|
/// let string = String::from_utf8(hello_vec);
|
|
|
|
|
/// assert_eq!(string, Ok("hello".to_string()));
|
|
|
|
|
/// ```
|
2014-04-10 20:55:34 +10:00
|
|
|
|
#[inline]
|
2014-05-22 16:57:53 -07:00
|
|
|
|
pub fn from_utf8(vec: Vec<u8>) -> Result<String, Vec<u8>> {
|
2014-04-10 20:55:34 +10:00
|
|
|
|
if str::is_utf8(vec.as_slice()) {
|
2014-05-22 16:57:53 -07:00
|
|
|
|
Ok(String { vec: vec })
|
2014-04-10 20:55:34 +10:00
|
|
|
|
} else {
|
2014-05-14 16:55:24 -07:00
|
|
|
|
Err(vec)
|
2014-04-10 20:55:34 +10:00
|
|
|
|
}
|
|
|
|
|
}
|
2014-07-10 17:43:03 +02:00
|
|
|
|
|
|
|
|
|
/// Decode a UTF-16 encoded vector `v` into a string, returning `None`
|
|
|
|
|
/// if `v` contains any invalid data.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```rust
|
|
|
|
|
/// // ð„žmusic
|
|
|
|
|
/// let mut v = [0xD834, 0xDD1E, 0x006d, 0x0075,
|
|
|
|
|
/// 0x0073, 0x0069, 0x0063];
|
|
|
|
|
/// assert_eq!(String::from_utf16(v), Some("ð„žmusic".to_string()));
|
|
|
|
|
///
|
|
|
|
|
/// // ð„žmu<invalid>ic
|
|
|
|
|
/// v[4] = 0xD800;
|
|
|
|
|
/// assert_eq!(String::from_utf16(v), None);
|
|
|
|
|
/// ```
|
|
|
|
|
pub fn from_utf16(v: &[u16]) -> Option<String> {
|
|
|
|
|
let mut s = String::with_capacity(v.len() / 2);
|
|
|
|
|
for c in str::utf16_items(v) {
|
|
|
|
|
match c {
|
|
|
|
|
str::ScalarValue(c) => s.push_char(c),
|
|
|
|
|
str::LoneSurrogate(_) => return None
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Some(s)
|
|
|
|
|
}
|
2014-07-10 17:53:51 +02:00
|
|
|
|
|
|
|
|
|
/// Decode a UTF-16 encoded vector `v` into a string, replacing
|
|
|
|
|
/// invalid data with the replacement character (U+FFFD).
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```rust
|
|
|
|
|
/// // ð„žmus<invalid>ic<invalid>
|
|
|
|
|
/// let v = [0xD834, 0xDD1E, 0x006d, 0x0075,
|
|
|
|
|
/// 0x0073, 0xDD1E, 0x0069, 0x0063,
|
|
|
|
|
/// 0xD834];
|
|
|
|
|
///
|
|
|
|
|
/// assert_eq!(String::from_utf16_lossy(v),
|
|
|
|
|
/// "ð„žmus\uFFFDic\uFFFD".to_string());
|
|
|
|
|
/// ```
|
|
|
|
|
pub fn from_utf16_lossy(v: &[u16]) -> String {
|
|
|
|
|
str::utf16_items(v).map(|c| c.to_char_lossy()).collect()
|
|
|
|
|
}
|
2014-07-04 22:18:11 +02:00
|
|
|
|
|
2014-07-04 21:55:58 +02:00
|
|
|
|
/// Convert a vector of chars to a string
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```rust
|
|
|
|
|
/// let chars = ['h', 'e', 'l', 'l', 'o'];
|
|
|
|
|
/// let string = String::from_chars(chars);
|
|
|
|
|
/// assert_eq!(string.as_slice(), "hello");
|
|
|
|
|
/// ```
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn from_chars(chs: &[char]) -> String {
|
|
|
|
|
chs.iter().map(|c| *c).collect()
|
|
|
|
|
}
|
2014-04-10 20:55:34 +10:00
|
|
|
|
|
|
|
|
|
/// Return the underlying byte buffer, encoded as UTF-8.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn into_bytes(self) -> Vec<u8> {
|
|
|
|
|
self.vec
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-02 16:54:22 -07:00
|
|
|
|
/// Pushes the given string onto this buffer; then, returns `self` so that it can be used
|
|
|
|
|
/// again.
|
|
|
|
|
#[inline]
|
2014-05-22 16:57:53 -07:00
|
|
|
|
pub fn append(mut self, second: &str) -> String {
|
2014-04-02 16:54:22 -07:00
|
|
|
|
self.push_str(second);
|
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Creates a string buffer by repeating a character `length` times.
|
|
|
|
|
#[inline]
|
2014-05-22 16:57:53 -07:00
|
|
|
|
pub fn from_char(length: uint, ch: char) -> String {
|
2014-04-02 16:54:22 -07:00
|
|
|
|
if length == 0 {
|
2014-05-22 16:57:53 -07:00
|
|
|
|
return String::new()
|
2014-04-02 16:54:22 -07:00
|
|
|
|
}
|
|
|
|
|
|
2014-05-22 16:57:53 -07:00
|
|
|
|
let mut buf = String::new();
|
2014-04-02 16:54:22 -07:00
|
|
|
|
buf.push_char(ch);
|
|
|
|
|
let size = buf.len() * length;
|
|
|
|
|
buf.reserve(size);
|
|
|
|
|
for _ in range(1, length) {
|
|
|
|
|
buf.push_char(ch)
|
|
|
|
|
}
|
|
|
|
|
buf
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-04 22:18:11 +02:00
|
|
|
|
/// Convert a byte to a UTF-8 string
|
|
|
|
|
///
|
|
|
|
|
/// # Failure
|
|
|
|
|
///
|
|
|
|
|
/// Fails if invalid UTF-8
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```rust
|
|
|
|
|
/// let string = String::from_byte(104);
|
|
|
|
|
/// assert_eq!(string.as_slice(), "h");
|
|
|
|
|
/// ```
|
|
|
|
|
pub fn from_byte(b: u8) -> String {
|
|
|
|
|
assert!(b < 128u8);
|
|
|
|
|
String::from_char(1, b as char)
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-02 16:54:22 -07:00
|
|
|
|
/// Pushes the given string onto this string buffer.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn push_str(&mut self, string: &str) {
|
|
|
|
|
self.vec.push_all(string.as_bytes())
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-10 20:55:34 +10:00
|
|
|
|
/// Push `ch` onto the given string `count` times.
|
2014-04-02 16:54:22 -07:00
|
|
|
|
#[inline]
|
|
|
|
|
pub fn grow(&mut self, count: uint, ch: char) {
|
|
|
|
|
for _ in range(0, count) {
|
|
|
|
|
self.push_char(ch)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Returns the number of bytes that this string buffer can hold without reallocating.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn byte_capacity(&self) -> uint {
|
|
|
|
|
self.vec.capacity()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Reserves capacity for at least `extra` additional bytes in this string buffer.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn reserve_additional(&mut self, extra: uint) {
|
|
|
|
|
self.vec.reserve_additional(extra)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Reserves capacity for at least `capacity` bytes in this string buffer.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn reserve(&mut self, capacity: uint) {
|
|
|
|
|
self.vec.reserve(capacity)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Reserves capacity for exactly `capacity` bytes in this string buffer.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn reserve_exact(&mut self, capacity: uint) {
|
|
|
|
|
self.vec.reserve_exact(capacity)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Shrinks the capacity of this string buffer to match its length.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn shrink_to_fit(&mut self) {
|
|
|
|
|
self.vec.shrink_to_fit()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Adds the given character to the end of the string.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn push_char(&mut self, ch: char) {
|
|
|
|
|
let cur_len = self.len();
|
std: Recreate a `collections` module
As with the previous commit with `librand`, this commit shuffles around some
`collections` code. The new state of the world is similar to that of librand:
* The libcollections crate now only depends on libcore and liballoc.
* The standard library has a new module, `std::collections`. All functionality
of libcollections is reexported through this module.
I would like to stress that this change is purely cosmetic. There are very few
alterations to these primitives.
There are a number of notable points about the new organization:
* std::{str, slice, string, vec} all moved to libcollections. There is no reason
that these primitives shouldn't be necessarily usable in a freestanding
context that has allocation. These are all reexported in their usual places in
the standard library.
* The `hashmap`, and transitively the `lru_cache`, modules no longer reside in
`libcollections`, but rather in libstd. The reason for this is because the
`HashMap::new` contructor requires access to the OSRng for initially seeding
the hash map. Beyond this requirement, there is no reason that the hashmap
could not move to libcollections.
I do, however, have a plan to move the hash map to the collections module. The
`HashMap::new` function could be altered to require that the `H` hasher
parameter ascribe to the `Default` trait, allowing the entire `hashmap` module
to live in libcollections. The key idea would be that the default hasher would
be different in libstd. Something along the lines of:
// src/libstd/collections/mod.rs
pub type HashMap<K, V, H = RandomizedSipHasher> =
core_collections::HashMap<K, V, H>;
This is not possible today because you cannot invoke static methods through
type aliases. If we modified the compiler, however, to allow invocation of
static methods through type aliases, then this type definition would
essentially be switching the default hasher from `SipHasher` in libcollections
to a libstd-defined `RandomizedSipHasher` type. This type's `Default`
implementation would randomly seed the `SipHasher` instance, and otherwise
perform the same as `SipHasher`.
This future state doesn't seem incredibly far off, but until that time comes,
the hashmap module will live in libstd to not compromise on functionality.
* In preparation for the hashmap moving to libcollections, the `hash` module has
moved from libstd to libcollections. A previously snapshotted commit enables a
distinct `Writer` trait to live in the `hash` module which `Hash`
implementations are now parameterized over.
Due to using a custom trait, the `SipHasher` implementation has lost its
specialized methods for writing integers. These can be re-added
backwards-compatibly in the future via default methods if necessary, but the
FNV hashing should satisfy much of the need for speedier hashing.
A list of breaking changes:
* HashMap::{get, get_mut} no longer fails with the key formatted into the error
message with `{:?}`, instead, a generic message is printed. With backtraces,
it should still be not-too-hard to track down errors.
* The HashMap, HashSet, and LruCache types are now available through
std::collections instead of the collections crate.
* Manual implementations of hash should be parameterized over `hash::Writer`
instead of just `Writer`.
[breaking-change]
2014-05-29 18:50:12 -07:00
|
|
|
|
// This may use up to 4 bytes.
|
|
|
|
|
self.vec.reserve_additional(4);
|
2014-04-02 16:54:22 -07:00
|
|
|
|
|
std: Recreate a `collections` module
As with the previous commit with `librand`, this commit shuffles around some
`collections` code. The new state of the world is similar to that of librand:
* The libcollections crate now only depends on libcore and liballoc.
* The standard library has a new module, `std::collections`. All functionality
of libcollections is reexported through this module.
I would like to stress that this change is purely cosmetic. There are very few
alterations to these primitives.
There are a number of notable points about the new organization:
* std::{str, slice, string, vec} all moved to libcollections. There is no reason
that these primitives shouldn't be necessarily usable in a freestanding
context that has allocation. These are all reexported in their usual places in
the standard library.
* The `hashmap`, and transitively the `lru_cache`, modules no longer reside in
`libcollections`, but rather in libstd. The reason for this is because the
`HashMap::new` contructor requires access to the OSRng for initially seeding
the hash map. Beyond this requirement, there is no reason that the hashmap
could not move to libcollections.
I do, however, have a plan to move the hash map to the collections module. The
`HashMap::new` function could be altered to require that the `H` hasher
parameter ascribe to the `Default` trait, allowing the entire `hashmap` module
to live in libcollections. The key idea would be that the default hasher would
be different in libstd. Something along the lines of:
// src/libstd/collections/mod.rs
pub type HashMap<K, V, H = RandomizedSipHasher> =
core_collections::HashMap<K, V, H>;
This is not possible today because you cannot invoke static methods through
type aliases. If we modified the compiler, however, to allow invocation of
static methods through type aliases, then this type definition would
essentially be switching the default hasher from `SipHasher` in libcollections
to a libstd-defined `RandomizedSipHasher` type. This type's `Default`
implementation would randomly seed the `SipHasher` instance, and otherwise
perform the same as `SipHasher`.
This future state doesn't seem incredibly far off, but until that time comes,
the hashmap module will live in libstd to not compromise on functionality.
* In preparation for the hashmap moving to libcollections, the `hash` module has
moved from libstd to libcollections. A previously snapshotted commit enables a
distinct `Writer` trait to live in the `hash` module which `Hash`
implementations are now parameterized over.
Due to using a custom trait, the `SipHasher` implementation has lost its
specialized methods for writing integers. These can be re-added
backwards-compatibly in the future via default methods if necessary, but the
FNV hashing should satisfy much of the need for speedier hashing.
A list of breaking changes:
* HashMap::{get, get_mut} no longer fails with the key formatted into the error
message with `{:?}`, instead, a generic message is printed. With backtraces,
it should still be not-too-hard to track down errors.
* The HashMap, HashSet, and LruCache types are now available through
std::collections instead of the collections crate.
* Manual implementations of hash should be parameterized over `hash::Writer`
instead of just `Writer`.
[breaking-change]
2014-05-29 18:50:12 -07:00
|
|
|
|
unsafe {
|
2014-04-02 16:54:22 -07:00
|
|
|
|
// Attempt to not use an intermediate buffer by just pushing bytes
|
|
|
|
|
// directly onto this string.
|
std: Recreate a `collections` module
As with the previous commit with `librand`, this commit shuffles around some
`collections` code. The new state of the world is similar to that of librand:
* The libcollections crate now only depends on libcore and liballoc.
* The standard library has a new module, `std::collections`. All functionality
of libcollections is reexported through this module.
I would like to stress that this change is purely cosmetic. There are very few
alterations to these primitives.
There are a number of notable points about the new organization:
* std::{str, slice, string, vec} all moved to libcollections. There is no reason
that these primitives shouldn't be necessarily usable in a freestanding
context that has allocation. These are all reexported in their usual places in
the standard library.
* The `hashmap`, and transitively the `lru_cache`, modules no longer reside in
`libcollections`, but rather in libstd. The reason for this is because the
`HashMap::new` contructor requires access to the OSRng for initially seeding
the hash map. Beyond this requirement, there is no reason that the hashmap
could not move to libcollections.
I do, however, have a plan to move the hash map to the collections module. The
`HashMap::new` function could be altered to require that the `H` hasher
parameter ascribe to the `Default` trait, allowing the entire `hashmap` module
to live in libcollections. The key idea would be that the default hasher would
be different in libstd. Something along the lines of:
// src/libstd/collections/mod.rs
pub type HashMap<K, V, H = RandomizedSipHasher> =
core_collections::HashMap<K, V, H>;
This is not possible today because you cannot invoke static methods through
type aliases. If we modified the compiler, however, to allow invocation of
static methods through type aliases, then this type definition would
essentially be switching the default hasher from `SipHasher` in libcollections
to a libstd-defined `RandomizedSipHasher` type. This type's `Default`
implementation would randomly seed the `SipHasher` instance, and otherwise
perform the same as `SipHasher`.
This future state doesn't seem incredibly far off, but until that time comes,
the hashmap module will live in libstd to not compromise on functionality.
* In preparation for the hashmap moving to libcollections, the `hash` module has
moved from libstd to libcollections. A previously snapshotted commit enables a
distinct `Writer` trait to live in the `hash` module which `Hash`
implementations are now parameterized over.
Due to using a custom trait, the `SipHasher` implementation has lost its
specialized methods for writing integers. These can be re-added
backwards-compatibly in the future via default methods if necessary, but the
FNV hashing should satisfy much of the need for speedier hashing.
A list of breaking changes:
* HashMap::{get, get_mut} no longer fails with the key formatted into the error
message with `{:?}`, instead, a generic message is printed. With backtraces,
it should still be not-too-hard to track down errors.
* The HashMap, HashSet, and LruCache types are now available through
std::collections instead of the collections crate.
* Manual implementations of hash should be parameterized over `hash::Writer`
instead of just `Writer`.
[breaking-change]
2014-05-29 18:50:12 -07:00
|
|
|
|
let slice = Slice {
|
|
|
|
|
data: self.vec.as_ptr().offset(cur_len as int),
|
|
|
|
|
len: 4,
|
|
|
|
|
};
|
|
|
|
|
let used = ch.encode_utf8(mem::transmute(slice));
|
2014-04-02 16:54:22 -07:00
|
|
|
|
self.vec.set_len(cur_len + used);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Pushes the given bytes onto this string buffer. This is unsafe because it does not check
|
|
|
|
|
/// to ensure that the resulting string will be valid UTF-8.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub unsafe fn push_bytes(&mut self, bytes: &[u8]) {
|
|
|
|
|
self.vec.push_all(bytes)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Works with the underlying buffer as a byte slice.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn as_bytes<'a>(&'a self) -> &'a [u8] {
|
|
|
|
|
self.vec.as_slice()
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-19 17:23:26 -07:00
|
|
|
|
/// Works with the underlying buffer as a mutable byte slice. Unsafe
|
|
|
|
|
/// because this can be used to violate the UTF-8 property.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub unsafe fn as_mut_bytes<'a>(&'a mut self) -> &'a mut [u8] {
|
|
|
|
|
self.vec.as_mut_slice()
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-02 16:54:22 -07:00
|
|
|
|
/// Shorten a string to the specified length (which must be <= the current length)
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn truncate(&mut self, len: uint) {
|
|
|
|
|
assert!(self.as_slice().is_char_boundary(len));
|
|
|
|
|
self.vec.truncate(len)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Appends a byte to this string buffer. The caller must preserve the valid UTF-8 property.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub unsafe fn push_byte(&mut self, byte: u8) {
|
2014-07-06 01:11:13 +01:00
|
|
|
|
self.vec.push(byte)
|
2014-04-02 16:54:22 -07:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Removes the last byte from the string buffer and returns it. Returns `None` if this string
|
|
|
|
|
/// buffer is empty.
|
|
|
|
|
///
|
|
|
|
|
/// The caller must preserve the valid UTF-8 property.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub unsafe fn pop_byte(&mut self) -> Option<u8> {
|
|
|
|
|
let len = self.len();
|
|
|
|
|
if len == 0 {
|
|
|
|
|
return None
|
|
|
|
|
}
|
|
|
|
|
|
2014-06-19 18:22:33 -07:00
|
|
|
|
let byte = self.as_bytes()[len - 1];
|
2014-04-02 16:54:22 -07:00
|
|
|
|
self.vec.set_len(len - 1);
|
|
|
|
|
Some(byte)
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-08 21:42:40 +01:00
|
|
|
|
/// Removes the last character from the string buffer and returns it. Returns `None` if this
|
|
|
|
|
/// string buffer is empty.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn pop_char(&mut self) -> Option<char> {
|
|
|
|
|
let len = self.len();
|
|
|
|
|
if len == 0 {
|
|
|
|
|
return None
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let CharRange {ch, next} = self.as_slice().char_range_at_reverse(len);
|
|
|
|
|
unsafe {
|
|
|
|
|
self.vec.set_len(next);
|
|
|
|
|
}
|
|
|
|
|
Some(ch)
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-02 16:54:22 -07:00
|
|
|
|
/// Removes the first byte from the string buffer and returns it. Returns `None` if this string
|
|
|
|
|
/// buffer is empty.
|
|
|
|
|
///
|
|
|
|
|
/// The caller must preserve the valid UTF-8 property.
|
|
|
|
|
pub unsafe fn shift_byte(&mut self) -> Option<u8> {
|
2014-05-08 21:42:40 +01:00
|
|
|
|
self.vec.shift()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Removes the first character from the string buffer and returns it. Returns `None` if this
|
|
|
|
|
/// string buffer is empty.
|
|
|
|
|
///
|
|
|
|
|
/// # Warning
|
|
|
|
|
///
|
|
|
|
|
/// This is a O(n) operation as it requires copying every element in the buffer.
|
|
|
|
|
pub fn shift_char (&mut self) -> Option<char> {
|
2014-04-02 16:54:22 -07:00
|
|
|
|
let len = self.len();
|
|
|
|
|
if len == 0 {
|
|
|
|
|
return None
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-08 21:42:40 +01:00
|
|
|
|
let CharRange {ch, next} = self.as_slice().char_range_at(0);
|
|
|
|
|
let new_len = len - next;
|
|
|
|
|
unsafe {
|
|
|
|
|
ptr::copy_memory(self.vec.as_mut_ptr(), self.vec.as_ptr().offset(next as int), new_len);
|
|
|
|
|
self.vec.set_len(new_len);
|
|
|
|
|
}
|
|
|
|
|
Some(ch)
|
2014-04-02 16:54:22 -07:00
|
|
|
|
}
|
2014-04-12 22:44:31 +10:00
|
|
|
|
|
|
|
|
|
/// Views the string buffer as a mutable sequence of bytes.
|
|
|
|
|
///
|
|
|
|
|
/// Callers must preserve the valid UTF-8 property.
|
|
|
|
|
pub unsafe fn as_mut_vec<'a>(&'a mut self) -> &'a mut Vec<u8> {
|
|
|
|
|
&mut self.vec
|
|
|
|
|
}
|
2014-04-02 16:54:22 -07:00
|
|
|
|
}
|
|
|
|
|
|
2014-05-19 11:32:09 -07:00
|
|
|
|
impl Collection for String {
|
2014-04-02 16:54:22 -07:00
|
|
|
|
#[inline]
|
|
|
|
|
fn len(&self) -> uint {
|
|
|
|
|
self.vec.len()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-22 16:57:53 -07:00
|
|
|
|
impl Mutable for String {
|
2014-05-11 03:49:09 -07:00
|
|
|
|
#[inline]
|
|
|
|
|
fn clear(&mut self) {
|
|
|
|
|
self.vec.clear()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-22 16:57:53 -07:00
|
|
|
|
impl FromIterator<char> for String {
|
|
|
|
|
fn from_iter<I:Iterator<char>>(iterator: I) -> String {
|
|
|
|
|
let mut buf = String::new();
|
2014-04-02 16:54:22 -07:00
|
|
|
|
buf.extend(iterator);
|
|
|
|
|
buf
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-22 16:57:53 -07:00
|
|
|
|
impl Extendable<char> for String {
|
2014-04-02 16:54:22 -07:00
|
|
|
|
fn extend<I:Iterator<char>>(&mut self, mut iterator: I) {
|
|
|
|
|
for ch in iterator {
|
|
|
|
|
self.push_char(ch)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-22 16:57:53 -07:00
|
|
|
|
impl Str for String {
|
2014-04-02 16:54:22 -07:00
|
|
|
|
#[inline]
|
|
|
|
|
fn as_slice<'a>(&'a self) -> &'a str {
|
|
|
|
|
unsafe {
|
core: Remove the cast module
This commit revisits the `cast` module in libcore and libstd, and scrutinizes
all functions inside of it. The result was to remove the `cast` module entirely,
folding all functionality into the `mem` module. Specifically, this is the fate
of each function in the `cast` module.
* transmute - This function was moved to `mem`, but it is now marked as
#[unstable]. This is due to planned changes to the `transmute`
function and how it can be invoked (see the #[unstable] comment).
For more information, see RFC 5 and #12898
* transmute_copy - This function was moved to `mem`, with clarification that is
is not an error to invoke it with T/U that are different
sizes, but rather that it is strongly discouraged. This
function is now #[stable]
* forget - This function was moved to `mem` and marked #[stable]
* bump_box_refcount - This function was removed due to the deprecation of
managed boxes as well as its questionable utility.
* transmute_mut - This function was previously deprecated, and removed as part
of this commit.
* transmute_mut_unsafe - This function doesn't serve much of a purpose when it
can be achieved with an `as` in safe code, so it was
removed.
* transmute_lifetime - This function was removed because it is likely a strong
indication that code is incorrect in the first place.
* transmute_mut_lifetime - This function was removed for the same reasons as
`transmute_lifetime`
* copy_lifetime - This function was moved to `mem`, but it is marked
`#[unstable]` now due to the likelihood of being removed in
the future if it is found to not be very useful.
* copy_mut_lifetime - This function was also moved to `mem`, but had the same
treatment as `copy_lifetime`.
* copy_lifetime_vec - This function was removed because it is not used today,
and its existence is not necessary with DST
(copy_lifetime will suffice).
In summary, the cast module was stripped down to these functions, and then the
functions were moved to the `mem` module.
transmute - #[unstable]
transmute_copy - #[stable]
forget - #[stable]
copy_lifetime - #[unstable]
copy_mut_lifetime - #[unstable]
[breaking-change]
2014-05-09 10:34:51 -07:00
|
|
|
|
mem::transmute(self.vec.as_slice())
|
2014-04-02 16:54:22 -07:00
|
|
|
|
}
|
|
|
|
|
}
|
2014-04-30 23:06:36 -07:00
|
|
|
|
}
|
2014-04-02 16:54:22 -07:00
|
|
|
|
|
2014-05-22 16:57:53 -07:00
|
|
|
|
impl StrAllocating for String {
|
2014-04-12 22:44:31 +10:00
|
|
|
|
#[inline]
|
2014-05-25 03:17:19 -07:00
|
|
|
|
fn into_string(self) -> String {
|
2014-05-19 17:23:26 -07:00
|
|
|
|
self
|
|
|
|
|
}
|
2014-04-02 16:54:22 -07:00
|
|
|
|
}
|
|
|
|
|
|
2014-05-22 16:57:53 -07:00
|
|
|
|
impl Default for String {
|
|
|
|
|
fn default() -> String {
|
|
|
|
|
String::new()
|
2014-05-19 23:19:56 -07:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-22 16:57:53 -07:00
|
|
|
|
impl fmt::Show for String {
|
2014-04-02 16:54:22 -07:00
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
|
self.as_slice().fmt(f)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
std: Recreate a `collections` module
As with the previous commit with `librand`, this commit shuffles around some
`collections` code. The new state of the world is similar to that of librand:
* The libcollections crate now only depends on libcore and liballoc.
* The standard library has a new module, `std::collections`. All functionality
of libcollections is reexported through this module.
I would like to stress that this change is purely cosmetic. There are very few
alterations to these primitives.
There are a number of notable points about the new organization:
* std::{str, slice, string, vec} all moved to libcollections. There is no reason
that these primitives shouldn't be necessarily usable in a freestanding
context that has allocation. These are all reexported in their usual places in
the standard library.
* The `hashmap`, and transitively the `lru_cache`, modules no longer reside in
`libcollections`, but rather in libstd. The reason for this is because the
`HashMap::new` contructor requires access to the OSRng for initially seeding
the hash map. Beyond this requirement, there is no reason that the hashmap
could not move to libcollections.
I do, however, have a plan to move the hash map to the collections module. The
`HashMap::new` function could be altered to require that the `H` hasher
parameter ascribe to the `Default` trait, allowing the entire `hashmap` module
to live in libcollections. The key idea would be that the default hasher would
be different in libstd. Something along the lines of:
// src/libstd/collections/mod.rs
pub type HashMap<K, V, H = RandomizedSipHasher> =
core_collections::HashMap<K, V, H>;
This is not possible today because you cannot invoke static methods through
type aliases. If we modified the compiler, however, to allow invocation of
static methods through type aliases, then this type definition would
essentially be switching the default hasher from `SipHasher` in libcollections
to a libstd-defined `RandomizedSipHasher` type. This type's `Default`
implementation would randomly seed the `SipHasher` instance, and otherwise
perform the same as `SipHasher`.
This future state doesn't seem incredibly far off, but until that time comes,
the hashmap module will live in libstd to not compromise on functionality.
* In preparation for the hashmap moving to libcollections, the `hash` module has
moved from libstd to libcollections. A previously snapshotted commit enables a
distinct `Writer` trait to live in the `hash` module which `Hash`
implementations are now parameterized over.
Due to using a custom trait, the `SipHasher` implementation has lost its
specialized methods for writing integers. These can be re-added
backwards-compatibly in the future via default methods if necessary, but the
FNV hashing should satisfy much of the need for speedier hashing.
A list of breaking changes:
* HashMap::{get, get_mut} no longer fails with the key formatted into the error
message with `{:?}`, instead, a generic message is printed. With backtraces,
it should still be not-too-hard to track down errors.
* The HashMap, HashSet, and LruCache types are now available through
std::collections instead of the collections crate.
* Manual implementations of hash should be parameterized over `hash::Writer`
instead of just `Writer`.
[breaking-change]
2014-05-29 18:50:12 -07:00
|
|
|
|
impl<H: hash::Writer> hash::Hash<H> for String {
|
2014-04-02 16:54:22 -07:00
|
|
|
|
#[inline]
|
|
|
|
|
fn hash(&self, hasher: &mut H) {
|
|
|
|
|
self.as_slice().hash(hasher)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-22 16:57:53 -07:00
|
|
|
|
impl<'a, S: Str> Equiv<S> for String {
|
2014-05-16 10:45:16 -07:00
|
|
|
|
#[inline]
|
|
|
|
|
fn equiv(&self, other: &S) -> bool {
|
|
|
|
|
self.as_slice() == other.as_slice()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-27 21:34:00 -07:00
|
|
|
|
impl<S: Str> Add<S, String> for String {
|
|
|
|
|
fn add(&self, other: &S) -> String {
|
2014-06-21 03:39:03 -07:00
|
|
|
|
let mut s = String::from_str(self.as_slice());
|
2014-05-27 21:34:00 -07:00
|
|
|
|
s.push_str(other.as_slice());
|
|
|
|
|
return s;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-02 16:54:22 -07:00
|
|
|
|
#[cfg(test)]
|
|
|
|
|
mod tests {
|
2014-05-29 19:03:06 -07:00
|
|
|
|
use std::prelude::*;
|
|
|
|
|
use test::Bencher;
|
|
|
|
|
|
2014-06-06 16:33:44 -07:00
|
|
|
|
use Mutable;
|
2014-07-10 17:53:51 +02:00
|
|
|
|
use str;
|
2014-04-02 16:54:22 -07:00
|
|
|
|
use str::{Str, StrSlice};
|
2014-05-22 16:57:53 -07:00
|
|
|
|
use super::String;
|
2014-04-02 16:54:22 -07:00
|
|
|
|
|
2014-06-21 03:39:03 -07:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_from_str() {
|
|
|
|
|
let owned: Option<::std::string::String> = from_str("string");
|
|
|
|
|
assert_eq!(owned.as_ref().map(|s| s.as_slice()), Some("string"));
|
|
|
|
|
}
|
2014-07-10 17:53:51 +02:00
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_from_utf16() {
|
|
|
|
|
let pairs =
|
|
|
|
|
[(String::from_str("ð…ðŒ¿ðŒ»ð†ðŒ¹ðŒ»ðŒ°\n"),
|
|
|
|
|
vec![0xd800_u16, 0xdf45_u16, 0xd800_u16, 0xdf3f_u16,
|
|
|
|
|
0xd800_u16, 0xdf3b_u16, 0xd800_u16, 0xdf46_u16,
|
|
|
|
|
0xd800_u16, 0xdf39_u16, 0xd800_u16, 0xdf3b_u16,
|
|
|
|
|
0xd800_u16, 0xdf30_u16, 0x000a_u16]),
|
|
|
|
|
|
|
|
|
|
(String::from_str("ð’ð‘‰ð®ð‘€ð²ð‘‹ ðð²ð‘\n"),
|
|
|
|
|
vec![0xd801_u16, 0xdc12_u16, 0xd801_u16,
|
|
|
|
|
0xdc49_u16, 0xd801_u16, 0xdc2e_u16, 0xd801_u16,
|
|
|
|
|
0xdc40_u16, 0xd801_u16, 0xdc32_u16, 0xd801_u16,
|
|
|
|
|
0xdc4b_u16, 0x0020_u16, 0xd801_u16, 0xdc0f_u16,
|
|
|
|
|
0xd801_u16, 0xdc32_u16, 0xd801_u16, 0xdc4d_u16,
|
|
|
|
|
0x000a_u16]),
|
|
|
|
|
|
|
|
|
|
(String::from_str("ðŒ€ðŒ–ðŒ‹ðŒ„ðŒ‘ðŒ‰Â·ðŒŒðŒ„ðŒ•ðŒ„ðŒ‹ðŒ‰ðŒ‘\n"),
|
|
|
|
|
vec![0xd800_u16, 0xdf00_u16, 0xd800_u16, 0xdf16_u16,
|
|
|
|
|
0xd800_u16, 0xdf0b_u16, 0xd800_u16, 0xdf04_u16,
|
|
|
|
|
0xd800_u16, 0xdf11_u16, 0xd800_u16, 0xdf09_u16,
|
|
|
|
|
0x00b7_u16, 0xd800_u16, 0xdf0c_u16, 0xd800_u16,
|
|
|
|
|
0xdf04_u16, 0xd800_u16, 0xdf15_u16, 0xd800_u16,
|
|
|
|
|
0xdf04_u16, 0xd800_u16, 0xdf0b_u16, 0xd800_u16,
|
|
|
|
|
0xdf09_u16, 0xd800_u16, 0xdf11_u16, 0x000a_u16 ]),
|
|
|
|
|
|
|
|
|
|
(String::from_str("ð’‹ð’˜ð’ˆð’‘ð’›ð’’ ð’•ð’“ ð’ˆð’šð’ ð’ð’œð’’ð’–ð’† ð’•ð’†\n"),
|
|
|
|
|
vec![0xd801_u16, 0xdc8b_u16, 0xd801_u16, 0xdc98_u16,
|
|
|
|
|
0xd801_u16, 0xdc88_u16, 0xd801_u16, 0xdc91_u16,
|
|
|
|
|
0xd801_u16, 0xdc9b_u16, 0xd801_u16, 0xdc92_u16,
|
|
|
|
|
0x0020_u16, 0xd801_u16, 0xdc95_u16, 0xd801_u16,
|
|
|
|
|
0xdc93_u16, 0x0020_u16, 0xd801_u16, 0xdc88_u16,
|
|
|
|
|
0xd801_u16, 0xdc9a_u16, 0xd801_u16, 0xdc8d_u16,
|
|
|
|
|
0x0020_u16, 0xd801_u16, 0xdc8f_u16, 0xd801_u16,
|
|
|
|
|
0xdc9c_u16, 0xd801_u16, 0xdc92_u16, 0xd801_u16,
|
|
|
|
|
0xdc96_u16, 0xd801_u16, 0xdc86_u16, 0x0020_u16,
|
|
|
|
|
0xd801_u16, 0xdc95_u16, 0xd801_u16, 0xdc86_u16,
|
|
|
|
|
0x000a_u16 ]),
|
|
|
|
|
// Issue #12318, even-numbered non-BMP planes
|
|
|
|
|
(String::from_str("\U00020000"),
|
|
|
|
|
vec![0xD840, 0xDC00])];
|
|
|
|
|
|
|
|
|
|
for p in pairs.iter() {
|
|
|
|
|
let (s, u) = (*p).clone();
|
|
|
|
|
let s_as_utf16 = s.as_slice().utf16_units().collect::<Vec<u16>>();
|
|
|
|
|
let u_as_string = String::from_utf16(u.as_slice()).unwrap();
|
|
|
|
|
|
|
|
|
|
assert!(str::is_utf16(u.as_slice()));
|
|
|
|
|
assert_eq!(s_as_utf16, u);
|
|
|
|
|
|
|
|
|
|
assert_eq!(u_as_string, s);
|
|
|
|
|
assert_eq!(String::from_utf16_lossy(u.as_slice()), s);
|
|
|
|
|
|
|
|
|
|
assert_eq!(String::from_utf16(s_as_utf16.as_slice()).unwrap(), s);
|
|
|
|
|
assert_eq!(u_as_string.as_slice().utf16_units().collect::<Vec<u16>>(), u);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_utf16_invalid() {
|
|
|
|
|
// completely positive cases tested above.
|
|
|
|
|
// lead + eof
|
|
|
|
|
assert_eq!(String::from_utf16([0xD800]), None);
|
|
|
|
|
// lead + lead
|
|
|
|
|
assert_eq!(String::from_utf16([0xD800, 0xD800]), None);
|
|
|
|
|
|
|
|
|
|
// isolated trail
|
|
|
|
|
assert_eq!(String::from_utf16([0x0061, 0xDC00]), None);
|
|
|
|
|
|
|
|
|
|
// general
|
|
|
|
|
assert_eq!(String::from_utf16([0xD800, 0xd801, 0xdc8b, 0xD800]), None);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_from_utf16_lossy() {
|
|
|
|
|
// completely positive cases tested above.
|
|
|
|
|
// lead + eof
|
|
|
|
|
assert_eq!(String::from_utf16_lossy([0xD800]), String::from_str("\uFFFD"));
|
|
|
|
|
// lead + lead
|
|
|
|
|
assert_eq!(String::from_utf16_lossy([0xD800, 0xD800]), String::from_str("\uFFFD\uFFFD"));
|
|
|
|
|
|
|
|
|
|
// isolated trail
|
|
|
|
|
assert_eq!(String::from_utf16_lossy([0x0061, 0xDC00]), String::from_str("a\uFFFD"));
|
|
|
|
|
|
|
|
|
|
// general
|
|
|
|
|
assert_eq!(String::from_utf16_lossy([0xD800, 0xd801, 0xdc8b, 0xD800]),
|
|
|
|
|
String::from_str("\uFFFDð’‹\uFFFD"));
|
|
|
|
|
}
|
2014-06-21 03:39:03 -07:00
|
|
|
|
|
2014-04-02 16:54:22 -07:00
|
|
|
|
#[bench]
|
2014-04-01 09:16:35 +08:00
|
|
|
|
fn bench_with_capacity(b: &mut Bencher) {
|
|
|
|
|
b.iter(|| {
|
2014-05-22 16:57:53 -07:00
|
|
|
|
String::with_capacity(100)
|
2014-04-02 16:54:22 -07:00
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[bench]
|
2014-04-01 09:16:35 +08:00
|
|
|
|
fn bench_push_str(b: &mut Bencher) {
|
2014-04-02 16:54:22 -07:00
|
|
|
|
let s = "ศไทย中华Việt Nam; Mary had a little lamb, Little lamb";
|
2014-04-01 09:16:35 +08:00
|
|
|
|
b.iter(|| {
|
2014-05-22 16:57:53 -07:00
|
|
|
|
let mut r = String::new();
|
2014-04-02 16:54:22 -07:00
|
|
|
|
r.push_str(s);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2014-06-30 16:41:30 +02:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_str_from_utf8() {
|
|
|
|
|
let xs = Vec::from_slice(b"hello");
|
|
|
|
|
assert_eq!(String::from_utf8(xs), Ok("hello".to_string()));
|
|
|
|
|
|
|
|
|
|
let xs = Vec::from_slice("ศไทยä¸åŽViệt Nam".as_bytes());
|
|
|
|
|
assert_eq!(String::from_utf8(xs), Ok("ศไทยä¸åŽViệt Nam".to_string()));
|
|
|
|
|
|
|
|
|
|
let xs = Vec::from_slice(b"hello\xFF");
|
|
|
|
|
assert_eq!(String::from_utf8(xs),
|
|
|
|
|
Err(Vec::from_slice(b"hello\xFF")));
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-02 16:54:22 -07:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_push_bytes() {
|
2014-05-22 16:57:53 -07:00
|
|
|
|
let mut s = String::from_str("ABC");
|
2014-04-02 16:54:22 -07:00
|
|
|
|
unsafe {
|
|
|
|
|
s.push_bytes([ 'D' as u8 ]);
|
|
|
|
|
}
|
|
|
|
|
assert_eq!(s.as_slice(), "ABCD");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_push_str() {
|
2014-05-22 16:57:53 -07:00
|
|
|
|
let mut s = String::new();
|
2014-04-02 16:54:22 -07:00
|
|
|
|
s.push_str("");
|
|
|
|
|
assert_eq!(s.as_slice().slice_from(0), "");
|
|
|
|
|
s.push_str("abc");
|
|
|
|
|
assert_eq!(s.as_slice().slice_from(0), "abc");
|
|
|
|
|
s.push_str("ประเทศไทย中华Việt Nam");
|
|
|
|
|
assert_eq!(s.as_slice().slice_from(0), "abcประเทศไทย中华Việt Nam");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_push_char() {
|
2014-05-22 16:57:53 -07:00
|
|
|
|
let mut data = String::from_str("ประเทศไทย中");
|
2014-04-02 16:54:22 -07:00
|
|
|
|
data.push_char('华');
|
|
|
|
|
data.push_char('b'); // 1 byte
|
|
|
|
|
data.push_char('¢'); // 2 byte
|
|
|
|
|
data.push_char('€'); // 3 byte
|
|
|
|
|
data.push_char('𤭢'); // 4 byte
|
|
|
|
|
assert_eq!(data.as_slice(), "ประเทศไทย中华b¢€𤭢");
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-08 21:42:40 +01:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_pop_char() {
|
2014-05-22 16:57:53 -07:00
|
|
|
|
let mut data = String::from_str("ประเทศไทย中华b¢€𤭢");
|
2014-05-08 21:42:40 +01:00
|
|
|
|
assert_eq!(data.pop_char().unwrap(), '𤭢'); // 4 bytes
|
|
|
|
|
assert_eq!(data.pop_char().unwrap(), '€'); // 3 bytes
|
|
|
|
|
assert_eq!(data.pop_char().unwrap(), '¢'); // 2 bytes
|
|
|
|
|
assert_eq!(data.pop_char().unwrap(), 'b'); // 1 bytes
|
|
|
|
|
assert_eq!(data.pop_char().unwrap(), '华');
|
|
|
|
|
assert_eq!(data.as_slice(), "ประเทศไทย中");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_shift_char() {
|
2014-05-22 16:57:53 -07:00
|
|
|
|
let mut data = String::from_str("𤭢€¢b华ประเทศไทย中");
|
2014-05-08 21:42:40 +01:00
|
|
|
|
assert_eq!(data.shift_char().unwrap(), '𤭢'); // 4 bytes
|
|
|
|
|
assert_eq!(data.shift_char().unwrap(), '€'); // 3 bytes
|
|
|
|
|
assert_eq!(data.shift_char().unwrap(), '¢'); // 2 bytes
|
|
|
|
|
assert_eq!(data.shift_char().unwrap(), 'b'); // 1 bytes
|
|
|
|
|
assert_eq!(data.shift_char().unwrap(), '华');
|
|
|
|
|
assert_eq!(data.as_slice(), "ประเทศไทย中");
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-02 16:54:22 -07:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_str_truncate() {
|
2014-05-22 16:57:53 -07:00
|
|
|
|
let mut s = String::from_str("12345");
|
2014-04-02 16:54:22 -07:00
|
|
|
|
s.truncate(5);
|
|
|
|
|
assert_eq!(s.as_slice(), "12345");
|
|
|
|
|
s.truncate(3);
|
|
|
|
|
assert_eq!(s.as_slice(), "123");
|
|
|
|
|
s.truncate(0);
|
|
|
|
|
assert_eq!(s.as_slice(), "");
|
|
|
|
|
|
2014-05-22 16:57:53 -07:00
|
|
|
|
let mut s = String::from_str("12345");
|
2014-04-02 16:54:22 -07:00
|
|
|
|
let p = s.as_slice().as_ptr();
|
|
|
|
|
s.truncate(3);
|
|
|
|
|
s.push_str("6");
|
|
|
|
|
let p_ = s.as_slice().as_ptr();
|
|
|
|
|
assert_eq!(p_, p);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
#[should_fail]
|
|
|
|
|
fn test_str_truncate_invalid_len() {
|
2014-05-22 16:57:53 -07:00
|
|
|
|
let mut s = String::from_str("12345");
|
2014-04-02 16:54:22 -07:00
|
|
|
|
s.truncate(6);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
#[should_fail]
|
|
|
|
|
fn test_str_truncate_split_codepoint() {
|
2014-05-22 16:57:53 -07:00
|
|
|
|
let mut s = String::from_str("\u00FC"); // ü
|
2014-04-02 16:54:22 -07:00
|
|
|
|
s.truncate(1);
|
|
|
|
|
}
|
2014-05-11 03:49:09 -07:00
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_str_clear() {
|
2014-05-22 16:57:53 -07:00
|
|
|
|
let mut s = String::from_str("12345");
|
2014-05-11 03:49:09 -07:00
|
|
|
|
s.clear();
|
|
|
|
|
assert_eq!(s.len(), 0);
|
|
|
|
|
assert_eq!(s.as_slice(), "");
|
|
|
|
|
}
|
2014-05-27 21:34:00 -07:00
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_str_add() {
|
|
|
|
|
let a = String::from_str("12345");
|
|
|
|
|
let b = a + "2";
|
|
|
|
|
let b = b + String::from_str("2");
|
|
|
|
|
assert_eq!(b.len(), 7);
|
|
|
|
|
assert_eq!(b.as_slice(), "1234522");
|
|
|
|
|
}
|
2014-04-02 16:54:22 -07:00
|
|
|
|
}
|