#[cfg_attr(rustfmt, rustfmt_skip)]
static RUST_KEYWORDS: &'static [&'static str] = &[
"as",
"async",
"await",
"break",
"crate",
"dyn",
"else",
"enum",
"extern",
"false",
"fn",
"for",
"if",
"impl",
"in",
"let",
"loop",
"match",
"mod",
"move",
"mut",
"pub",
"ref",
"return",
"static",
"self",
"Self",
"struct",
"super",
"true",
"trait",
"type",
"unsafe",
"use",
"while",
"continue",
"box",
"const",
"where",
"virtual",
"proc",
"alignof",
"become",
"offsetof",
"priv",
"pure",
"sizeof",
"typeof",
"unsized",
"yield",
"do",
"abstract",
"final",
"override",
"macro",
];
pub fn is_rust_keyword(ident: &str) -> bool {
RUST_KEYWORDS.contains(&ident)
}
fn hex_digit(value: u32) -> char {
if value < 10 {
(b'0' + value as u8) as char
} else if value < 0x10 {
(b'a' + value as u8 - 10) as char
} else {
unreachable!()
}
}
pub fn quote_escape_str(s: &str) -> String {
let mut buf = String::new();
buf.push('"');
buf.extend(s.chars().flat_map(|c| c.escape_default()));
buf.push('"');
buf
}
pub fn quote_escape_bytes(bytes: &[u8]) -> String {
let mut buf = String::new();
buf.push('b');
buf.push('"');
for &b in bytes {
match b {
b'\n' => buf.push_str(r"\n"),
b'\r' => buf.push_str(r"\r"),
b'\t' => buf.push_str(r"\t"),
b'"' => buf.push_str("\\\""),
b'\\' => buf.push_str(r"\\"),
b'\x20'..=b'\x7e' => buf.push(b as char),
_ => {
buf.push_str(r"\x");
buf.push(hex_digit((b as u32) >> 4));
buf.push(hex_digit((b as u32) & 0x0f));
}
}
}
buf.push('"');
buf
}
#[cfg(test)]
mod test {
use super::*;
#[test]
fn test_quote_escape_bytes() {
assert_eq!("b\"\"", quote_escape_bytes(b""));
assert_eq!("b\"xyZW\"", quote_escape_bytes(b"xyZW"));
assert_eq!("b\"aa\\\"bb\"", quote_escape_bytes(b"aa\"bb"));
assert_eq!("b\"aa\\r\\n\\tbb\"", quote_escape_bytes(b"aa\r\n\tbb"));
assert_eq!(
"b\"\\x00\\x01\\x12\\xfe\\xff\"",
quote_escape_bytes(b"\x00\x01\x12\xfe\xff")
);
}
}