From 35653b04c9d6838f596c55356aab1ee9cd36702f Mon Sep 17 00:00:00 2001 From: Mads Marquart Date: Thu, 17 Feb 2022 13:58:07 +0100 Subject: [PATCH 1/5] Make form_urlencoded no_std compatible --- form_urlencoded/src/lib.rs | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/form_urlencoded/src/lib.rs b/form_urlencoded/src/lib.rs index b0916e42b..a952b7007 100644 --- a/form_urlencoded/src/lib.rs +++ b/form_urlencoded/src/lib.rs @@ -12,10 +12,14 @@ //! //! Converts between a string (such as an URL’s query string) //! and a sequence of (name, value) pairs. +#![no_std] +extern crate alloc; + +use alloc::borrow::{Borrow, Cow, ToOwned}; +use alloc::string::String; +use core::str; use percent_encoding::{percent_decode, percent_encode_byte}; -use std::borrow::{Borrow, Cow}; -use std::str; /// Convert a byte string in the `application/x-www-form-urlencoded` syntax /// into a iterator of (name, value) pairs. From 194122c1dbbd31bfb560ec3aa33ed2925d310b8a Mon Sep 17 00:00:00 2001 From: Mads Marquart Date: Thu, 12 Aug 2021 19:08:53 +0200 Subject: [PATCH 2/5] Make data-url no_std compatible --- data-url/src/forgiving_base64.rs | 2 ++ data-url/src/lib.rs | 6 ++++++ data-url/src/mime.rs | 5 +++-- 3 files changed, 11 insertions(+), 2 deletions(-) diff --git a/data-url/src/forgiving_base64.rs b/data-url/src/forgiving_base64.rs index 4f8550ed2..390d88bc4 100644 --- a/data-url/src/forgiving_base64.rs +++ b/data-url/src/forgiving_base64.rs @@ -1,5 +1,7 @@ //! +use alloc::vec::Vec; + #[derive(Debug)] pub struct InvalidBase64(InvalidBase64Details); diff --git a/data-url/src/lib.rs b/data-url/src/lib.rs index 4a4f376f4..32a14c442 100644 --- a/data-url/src/lib.rs +++ b/data-url/src/lib.rs @@ -14,6 +14,12 @@ //! assert_eq!(body, b"Hello World!"); //! assert!(fragment.is_none()); //! ``` +#![no_std] + +#[macro_use] +extern crate alloc; + +use alloc::{string::String, vec::Vec}; macro_rules! require { ($condition: expr) => { diff --git a/data-url/src/mime.rs b/data-url/src/mime.rs index a4c4f5aa4..baca4582f 100644 --- a/data-url/src/mime.rs +++ b/data-url/src/mime.rs @@ -1,5 +1,6 @@ -use std::fmt::{self, Write}; -use std::str::FromStr; +use alloc::{borrow::ToOwned, string::String, vec::Vec}; +use core::fmt::{self, Write}; +use core::str::FromStr; /// #[derive(Debug, PartialEq, Eq)] From 422a260f3806d827bc1d41880e4e8784c8bb4a45 Mon Sep 17 00:00:00 2001 From: Mads Marquart Date: Thu, 17 Feb 2022 14:10:28 +0100 Subject: [PATCH 3/5] Make idna no_std compatible Add default feature flag "std" that enables a `std::error::Error` impl for `Errors`. --- idna/Cargo.toml | 8 ++++++-- idna/src/lib.rs | 8 ++++++++ idna/src/punycode.rs | 7 ++++--- idna/src/uts46.rs | 11 +++++++---- 4 files changed, 25 insertions(+), 9 deletions(-) diff --git a/idna/Cargo.toml b/idna/Cargo.toml index 9566cc72e..7caa44595 100644 --- a/idna/Cargo.toml +++ b/idna/Cargo.toml @@ -12,6 +12,10 @@ rust-version = "1.45" [lib] doctest = false +[features] +default = ["std"] +std = [] + [[test]] name = "tests" harness = false @@ -26,8 +30,8 @@ tester = "0.9" serde_json = "1.0" [dependencies] -unicode-bidi = "0.3" -unicode-normalization = "0.1.17" +unicode-bidi = { version = "0.3.7", default-features = false } +unicode-normalization = { version = "0.1.17", default-features = false } [[bench]] name = "all" diff --git a/idna/src/lib.rs b/idna/src/lib.rs index 37d638741..a4d03c08a 100644 --- a/idna/src/lib.rs +++ b/idna/src/lib.rs @@ -31,11 +31,19 @@ //! > This document specifies a mechanism //! > that minimizes the impact of this transition for client software, //! > allowing client software to access domains that are valid under either system. +#![no_std] + +extern crate alloc; + +#[cfg(feature = "std")] +extern crate std; #[cfg(test)] #[macro_use] extern crate assert_matches; +use alloc::string::String; + pub mod punycode; mod uts46; diff --git a/idna/src/punycode.rs b/idna/src/punycode.rs index 21955f359..faef379ca 100644 --- a/idna/src/punycode.rs +++ b/idna/src/punycode.rs @@ -13,8 +13,9 @@ //! `encode_str` and `decode_to_string` provide convenience wrappers //! that convert from and to Rust’s UTF-8 based `str` and `String` types. -use std::char; -use std::u32; +use alloc::{string::String, vec::Vec}; +use core::char; +use core::u32; // Bootstring parameters for Punycode static BASE: u32 = 36; @@ -168,7 +169,7 @@ impl Decoder { } pub(crate) struct Decode<'a> { - base: std::str::Chars<'a>, + base: core::str::Chars<'a>, pub(crate) insertions: &'a [(usize, char)], inserted: usize, position: usize, diff --git a/idna/src/uts46.rs b/idna/src/uts46.rs index 9ee83bc88..6c7d82cd4 100644 --- a/idna/src/uts46.rs +++ b/idna/src/uts46.rs @@ -11,7 +11,9 @@ use self::Mapping::*; use crate::punycode; -use std::{error::Error as StdError, fmt}; + +use alloc::string::String; +use core::fmt; use unicode_bidi::{bidi_class, BidiClass}; use unicode_normalization::char::is_combining_mark; use unicode_normalization::{is_nfc, UnicodeNormalization}; @@ -70,10 +72,10 @@ fn find_char(codepoint: char) -> &'static Mapping { } struct Mapper<'a> { - chars: std::str::Chars<'a>, + chars: core::str::Chars<'a>, config: Config, errors: &'a mut Errors, - slice: Option>, + slice: Option>, } impl<'a> Iterator for Mapper<'a> { @@ -697,7 +699,8 @@ impl From for Result<(), Errors> { } } -impl StdError for Errors {} +#[cfg(feature = "std")] +impl std::error::Error for Errors {} impl fmt::Display for Errors { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { From 81ee0741aad668d4028521225f75e3a2de5f7c72 Mon Sep 17 00:00:00 2001 From: Mads Marquart Date: Tue, 5 Oct 2021 14:36:13 +0200 Subject: [PATCH 4/5] Test no_std support --- .github/workflows/main.yml | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/.github/workflows/main.yml b/.github/workflows/main.yml index 74b4c1610..e6cff540c 100644 --- a/.github/workflows/main.yml +++ b/.github/workflows/main.yml @@ -42,6 +42,12 @@ jobs: with: command: test args: --all-features + - uses: actions-rs/cargo@v1 + # --no-default-features only available since 1.51.0 + if: matrix.rust != '1.45.0' + with: + command: test + args: --no-default-features WASM: runs-on: ubuntu-latest From 67d2a4c49d2f1414fc2aa614e29c9f7e2b4cb5f8 Mon Sep 17 00:00:00 2001 From: Mads Marquart Date: Thu, 12 Aug 2021 19:12:51 +0200 Subject: [PATCH 5/5] Initial work at making url no_std compatible --- url/Cargo.toml | 7 ++-- url/src/host.rs | 7 ++-- url/src/lib.rs | 70 +++++++++++++++++++++++++++------------- url/src/origin.rs | 5 ++- url/src/parser.rs | 14 ++++---- url/src/path_segments.rs | 3 +- url/src/quirks.rs | 2 ++ url/src/slicing.rs | 3 +- 8 files changed, 75 insertions(+), 36 deletions(-) diff --git a/url/Cargo.toml b/url/Cargo.toml index 3955078b5..e49f42fce 100644 --- a/url/Cargo.toml +++ b/url/Cargo.toml @@ -26,12 +26,13 @@ bencher = "0.1" [dependencies] form_urlencoded = { version = "1.0.0", path = "../form_urlencoded" } -idna = { version = "0.2.0", path = "../idna", optional = true } +idna = { version = "0.2.0", default-features = false, path = "../idna", optional = true } percent-encoding = { version = "2.1.0", path = "../percent_encoding" } -serde = {version = "1.0", optional = true, features = ["derive"]} +serde = { version = "1.0", default-features = false, optional = true, features = ["derive"] } [features] -default = ["idna"] +default = ["std", "idna"] +std = ["idna/std"] [[bench]] name = "parse_url" diff --git a/url/src/host.rs b/url/src/host.rs index 4678cb8a1..df0890fec 100644 --- a/url/src/host.rs +++ b/url/src/host.rs @@ -6,8 +6,11 @@ // option. This file may not be copied, modified, or distributed // except according to those terms. -use std::cmp; -use std::fmt::{self, Formatter}; +use alloc::borrow::ToOwned; +use alloc::string::{String, ToString}; +use alloc::vec::Vec; +use core::cmp; +use core::fmt::{self, Formatter}; use std::net::{Ipv4Addr, Ipv6Addr}; use percent_encoding::{percent_decode, utf8_percent_encode, CONTROLS}; diff --git a/url/src/lib.rs b/url/src/lib.rs index 8bdd378b7..b0555f430 100644 --- a/url/src/lib.rs +++ b/url/src/lib.rs @@ -131,6 +131,10 @@ url = { version = "2", default-features = false } */ #![doc(html_root_url = "https://docs.rs/url/2.2.2")] +#![no_std] +#[macro_use] +extern crate alloc; +extern crate std; pub use form_urlencoded; @@ -138,20 +142,25 @@ pub use form_urlencoded; extern crate serde; use crate::host::HostInternal; -use crate::parser::{to_u32, Context, Parser, SchemeType, PATH_SEGMENT, USERINFO}; -use percent_encoding::{percent_decode, percent_encode, utf8_percent_encode}; -use std::borrow::Borrow; -use std::cmp; -use std::fmt::{self, Write}; -use std::hash; -use std::io; -use std::mem; -use std::net::{IpAddr, SocketAddr, ToSocketAddrs}; -use std::ops::{Range, RangeFrom, RangeTo}; -use std::path::{Path, PathBuf}; -use std::str; - -use std::convert::TryFrom; +use crate::parser::{to_u32, Context, Parser, SchemeType, USERINFO}; +use alloc::borrow::ToOwned; +use alloc::string::{String, ToString}; +use core::borrow::Borrow; +use core::cmp; +use core::convert::TryFrom; +use core::fmt::{self, Write}; +use core::hash; +use core::mem; +use core::ops::{Range, RangeFrom, RangeTo}; +use core::str; +use percent_encoding::utf8_percent_encode; +use std::net::IpAddr; +#[cfg(feature = "std")] +use std::{ + io, + net::{SocketAddr, ToSocketAddrs}, + path::{Path, PathBuf}, +}; pub use crate::host::Host; pub use crate::origin::{OpaqueOrigin, Origin}; @@ -1144,10 +1153,11 @@ impl Url { /// }) /// } /// ``` + #[cfg(feature = "std")] pub fn socket_addrs( &self, default_port_number: impl Fn() -> Option, - ) -> io::Result> { + ) -> io::Result> { // Note: trying to avoid the Vec allocation by returning `impl AsRef<[SocketAddr]>` // causes borrowck issues because the return value borrows `default_port_number`: // @@ -1156,6 +1166,7 @@ impl Url { // > This RFC proposes that *all* type parameters are considered in scope // > for `impl Trait` in return position + // TODO: Return custom error type to support no_std fn io_result(opt: Option, message: &str) -> io::Result { opt.ok_or_else(|| io::Error::new(io::ErrorKind::InvalidData, message)) } @@ -2314,7 +2325,9 @@ impl Url { /// # run().unwrap(); /// # } /// ``` - #[cfg(any(unix, windows, target_os = "redox", target_os = "wasi"))] + /// + /// This method is only available if the `std` Cargo feature is enabled. + #[cfg(all(feature = "std", any(unix, windows, target_os = "redox", target_os = "wasi")))] #[allow(clippy::result_unit_err)] pub fn from_file_path>(path: P) -> Result { let mut serialization = "file://".to_owned(); @@ -2351,7 +2364,9 @@ impl Url { /// /// Note that `std::path` does not consider trailing slashes significant /// and usually does not include them (e.g. in `Path::parent()`). - #[cfg(any(unix, windows, target_os = "redox", target_os = "wasi"))] + /// + /// This method is only available if the `std` Cargo feature is enabled. + #[cfg(all(feature = "std", any(unix, windows, target_os = "redox", target_os = "wasi")))] #[allow(clippy::result_unit_err)] pub fn from_directory_path>(path: P) -> Result { let mut url = Url::from_file_path(path)?; @@ -2467,8 +2482,10 @@ impl Url { /// or if `Path::new_opt()` returns `None`. /// (That is, if the percent-decoded path contains a NUL byte or, /// for a Windows path, is not UTF-8.) + /// + /// This method is only available if the `std` Cargo feature is enabled. #[inline] - #[cfg(any(unix, windows, target_os = "redox", target_os = "wasi"))] + #[cfg(all(feature = "std", any(unix, windows, target_os = "redox", target_os = "wasi")))] #[allow(clippy::result_unit_err)] pub fn to_file_path(&self) -> Result { if let Some(segments) = self.path_segments() { @@ -2672,11 +2689,13 @@ impl<'de> serde::Deserialize<'de> for Url { } } -#[cfg(any(unix, target_os = "redox", target_os = "wasi"))] +#[cfg(all(feature = "std", any(unix, target_os = "redox", target_os = "wasi")))] fn path_to_file_url_segments( path: &Path, serialization: &mut String, ) -> Result<(u32, HostInternal), ()> { + use crate::parser::PATH_SEGMENT; + use percent_encoding::percent_encode; #[cfg(any(unix, target_os = "redox"))] use std::os::unix::prelude::OsStrExt; #[cfg(target_os = "wasi")] @@ -2702,7 +2721,7 @@ fn path_to_file_url_segments( Ok((host_end, HostInternal::None)) } -#[cfg(windows)] +#[cfg(all(feature = "std", windows))] fn path_to_file_url_segments( path: &Path, serialization: &mut String, @@ -2710,12 +2729,15 @@ fn path_to_file_url_segments( path_to_file_url_segments_windows(path, serialization) } +#[cfg(feature = "std")] // Build this unconditionally to alleviate https://github.com/servo/rust-url/issues/102 #[cfg_attr(not(windows), allow(dead_code))] fn path_to_file_url_segments_windows( path: &Path, serialization: &mut String, ) -> Result<(u32, HostInternal), ()> { + use crate::parser::PATH_SEGMENT; + use percent_encoding::percent_encode; use std::path::{Component, Prefix}; if !path.is_absolute() { return Err(()); @@ -2770,11 +2792,13 @@ fn path_to_file_url_segments_windows( Ok((host_end, host_internal)) } -#[cfg(any(unix, target_os = "redox", target_os = "wasi"))] +#[cfg(all(feature = "std", any(unix, target_os = "redox", target_os = "wasi")))] fn file_url_segments_to_pathbuf( host: Option<&str>, segments: str::Split<'_, char>, ) -> Result { + use alloc::vec::Vec; + use percent_encoding::percent_decode; use std::ffi::OsStr; #[cfg(any(unix, target_os = "redox"))] use std::os::unix::prelude::OsStrExt; @@ -2810,7 +2834,7 @@ fn file_url_segments_to_pathbuf( Ok(path) } -#[cfg(windows)] +#[cfg(all(feature = "std", windows))] fn file_url_segments_to_pathbuf( host: Option<&str>, segments: str::Split, @@ -2818,12 +2842,14 @@ fn file_url_segments_to_pathbuf( file_url_segments_to_pathbuf_windows(host, segments) } +#[cfg(feature = "std")] // Build this unconditionally to alleviate https://github.com/servo/rust-url/issues/102 #[cfg_attr(not(windows), allow(dead_code))] fn file_url_segments_to_pathbuf_windows( host: Option<&str>, mut segments: str::Split<'_, char>, ) -> Result { + use percent_encoding::percent_decode; let mut string = if let Some(host) = host { r"\\".to_owned() + host } else { diff --git a/url/src/origin.rs b/url/src/origin.rs index 838be55bd..c3d64fe00 100644 --- a/url/src/origin.rs +++ b/url/src/origin.rs @@ -9,7 +9,10 @@ use crate::host::Host; use crate::parser::default_port; use crate::Url; -use std::sync::atomic::{AtomicUsize, Ordering}; +use alloc::borrow::ToOwned; +use alloc::string::String; +use core::sync::atomic::{AtomicUsize, Ordering}; +use idna::domain_to_unicode; pub fn url_origin(url: &Url) -> Origin { let scheme = url.scheme(); diff --git a/url/src/parser.rs b/url/src/parser.rs index ed4ca3c4e..4b8df3a6b 100644 --- a/url/src/parser.rs +++ b/url/src/parser.rs @@ -6,9 +6,10 @@ // option. This file may not be copied, modified, or distributed // except according to those terms. -use std::error::Error; -use std::fmt::{self, Formatter, Write}; -use std::str; +use alloc::borrow::ToOwned; +use alloc::string::{String, ToString}; +use core::fmt::{self, Formatter, Write}; +use core::str; use crate::host::{Host, HostInternal}; use crate::Url; @@ -72,7 +73,8 @@ macro_rules! simple_enum_error { } } -impl Error for ParseError {} +#[cfg(feature = "std")] +impl std::error::Error for ParseError {} simple_enum_error! { EmptyHost => "empty host", @@ -1107,7 +1109,7 @@ impl<'a> Parser<'a> { while let (Some(c), remaining) = input.split_first() { if let Some(digit) = c.to_digit(10) { port = port * 10 + digit; - if port > ::std::u16::MAX as u32 { + if port > core::u16::MAX as u32 { return Err(ParseError::InvalidPort); } has_any_digit = true; @@ -1541,7 +1543,7 @@ pub fn ascii_alpha(ch: char) -> bool { #[inline] pub fn to_u32(i: usize) -> ParseResult { - if i <= ::std::u32::MAX as usize { + if i <= core::u32::MAX as usize { Ok(i as u32) } else { Err(ParseError::Overflow) diff --git a/url/src/path_segments.rs b/url/src/path_segments.rs index 29afc1e7e..131f0b82d 100644 --- a/url/src/path_segments.rs +++ b/url/src/path_segments.rs @@ -8,7 +8,8 @@ use crate::parser::{self, to_u32, SchemeType}; use crate::Url; -use std::str; +use alloc::string::String; +use core::str; /// Exposes methods to manipulate the path of an URL that is not cannot-be-base. /// diff --git a/url/src/quirks.rs b/url/src/quirks.rs index 3ecb3c24d..6ec774657 100644 --- a/url/src/quirks.rs +++ b/url/src/quirks.rs @@ -11,6 +11,8 @@ //! Unless you need to be interoperable with web browsers, //! you probably want to use `Url` method instead. +use alloc::string::{String, ToString}; + use crate::parser::{default_port, Context, Input, Parser, SchemeType}; use crate::{Host, ParseError, Position, Url}; diff --git a/url/src/slicing.rs b/url/src/slicing.rs index a90337bb6..1c6fe7616 100644 --- a/url/src/slicing.rs +++ b/url/src/slicing.rs @@ -6,8 +6,9 @@ // option. This file may not be copied, modified, or distributed // except according to those terms. +use core::ops::{Index, Range, RangeFrom, RangeFull, RangeTo}; + use crate::Url; -use std::ops::{Index, Range, RangeFrom, RangeFull, RangeTo}; impl Index for Url { type Output = str;