rust-analyzer/crates/assists/src/handlers/raw_string.rs

505 lines
11 KiB
Rust
Raw Normal View History

use std::borrow::Cow;
2020-08-12 16:26:51 +00:00
use syntax::{
ast::{self, HasQuotes, HasStringValue},
2020-02-27 16:19:53 +00:00
AstToken,
2019-10-27 08:32:39 +00:00
SyntaxKind::{RAW_STRING, STRING},
TextRange, TextSize,
2019-10-27 08:32:39 +00:00
};
use test_utils::mark;
2019-09-20 15:55:59 +00:00
2020-06-28 22:36:05 +00:00
use crate::{AssistContext, AssistId, AssistKind, Assists};
2019-09-20 15:55:59 +00:00
2019-10-27 09:22:53 +00:00
// Assist: make_raw_string
//
// Adds `r#` to a plain string literal.
//
// ```
// fn main() {
// "Hello,<|> World!";
// }
// ```
// ->
// ```
// fn main() {
// r#"Hello, World!"#;
// }
// ```
pub(crate) fn make_raw_string(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
let token = ctx.find_token_at_offset(STRING).and_then(ast::String::cast)?;
let value = token.value()?;
let target = token.syntax().text_range();
2020-06-28 22:36:05 +00:00
acc.add(
2020-07-02 21:48:35 +00:00
AssistId("make_raw_string", AssistKind::RefactorRewrite),
2020-06-28 22:36:05 +00:00
"Rewrite as raw string",
target,
|edit| {
let hashes = "#".repeat(required_hashes(&value).max(1));
if matches!(value, Cow::Borrowed(_)) {
// Avoid replacing the whole string to better position the cursor.
edit.insert(token.syntax().text_range().start(), format!("r{}", hashes));
edit.insert(token.syntax().text_range().end(), format!("{}", hashes));
} else {
edit.replace(
token.syntax().text_range(),
format!("r{}\"{}\"{}", hashes, value, hashes),
);
2020-06-28 22:36:05 +00:00
}
},
)
2019-09-20 15:55:59 +00:00
}
2019-10-27 09:22:53 +00:00
// Assist: make_usual_string
//
// Turns a raw string into a plain string.
//
// ```
// fn main() {
// r#"Hello,<|> "World!""#;
// }
// ```
// ->
// ```
// fn main() {
// "Hello, \"World!\"";
// }
// ```
pub(crate) fn make_usual_string(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
let token = ctx.find_token_at_offset(RAW_STRING).and_then(ast::RawString::cast)?;
let value = token.value()?;
let target = token.syntax().text_range();
2020-06-28 22:36:05 +00:00
acc.add(
2020-07-02 21:48:35 +00:00
AssistId("make_usual_string", AssistKind::RefactorRewrite),
2020-06-28 22:36:05 +00:00
"Rewrite as regular string",
target,
|edit| {
// parse inside string to escape `"`
let escaped = value.escape_default().to_string();
if let Some(offsets) = token.quote_offsets() {
if token.text()[offsets.contents - token.syntax().text_range().start()] == escaped {
edit.replace(offsets.quotes.0, "\"");
edit.replace(offsets.quotes.1, "\"");
return;
}
}
2020-06-28 22:36:05 +00:00
edit.replace(token.syntax().text_range(), format!("\"{}\"", escaped));
},
)
2019-09-20 15:55:59 +00:00
}
2019-10-27 09:22:53 +00:00
// Assist: add_hash
//
// Adds a hash to a raw string literal.
//
// ```
// fn main() {
// r#"Hello,<|> World!"#;
// }
// ```
// ->
// ```
// fn main() {
// r##"Hello, World!"##;
// }
// ```
pub(crate) fn add_hash(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
let token = ctx.find_token_at_offset(RAW_STRING)?;
let target = token.text_range();
acc.add(AssistId("add_hash", AssistKind::Refactor), "Add #", target, |edit| {
2020-04-24 21:40:41 +00:00
edit.insert(token.text_range().start() + TextSize::of('r'), "#");
2019-10-27 08:32:39 +00:00
edit.insert(token.text_range().end(), "#");
})
2019-09-20 15:55:59 +00:00
}
2019-10-27 09:22:53 +00:00
// Assist: remove_hash
//
// Removes a hash from a raw string literal.
//
// ```
// fn main() {
// r#"Hello,<|> World!"#;
// }
// ```
// ->
// ```
// fn main() {
// r"Hello, World!";
// }
// ```
pub(crate) fn remove_hash(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
let token = ctx.find_token_at_offset(RAW_STRING).and_then(ast::RawString::cast)?;
2019-09-20 15:55:59 +00:00
let text = token.text().as_str();
2020-07-19 18:26:24 +00:00
if !text.starts_with("r#") && text.ends_with('#') {
2019-09-20 15:55:59 +00:00
return None;
}
let existing_hashes = text.chars().skip(1).take_while(|&it| it == '#').count();
let text_range = token.syntax().text_range();
let internal_text = &text[token.text_range_between_quotes()? - text_range.start()];
if existing_hashes == required_hashes(internal_text) {
mark::hit!(cant_remove_required_hash);
return None;
}
acc.add(AssistId("remove_hash", AssistKind::RefactorRewrite), "Remove #", text_range, |edit| {
edit.delete(TextRange::at(text_range.start() + TextSize::of('r'), TextSize::of('#')));
edit.delete(TextRange::new(text_range.end() - TextSize::of('#'), text_range.end()));
})
2019-09-20 15:55:59 +00:00
}
fn required_hashes(s: &str) -> usize {
let mut res = 0usize;
for idx in s.match_indices('"').map(|(i, _)| i) {
2019-10-27 08:32:39 +00:00
let (_, sub) = s.split_at(idx + 1);
let n_hashes = sub.chars().take_while(|c| *c == '#').count();
res = res.max(n_hashes + 1)
2019-10-27 08:32:39 +00:00
}
res
}
#[test]
fn test_required_hashes() {
assert_eq!(0, required_hashes("abc"));
assert_eq!(0, required_hashes("###"));
assert_eq!(1, required_hashes("\""));
assert_eq!(2, required_hashes("\"#abc"));
assert_eq!(0, required_hashes("#abc"));
assert_eq!(3, required_hashes("#ab\"##c"));
assert_eq!(5, required_hashes("#ab\"##\"####c"));
2019-10-27 08:32:39 +00:00
}
2019-09-20 15:55:59 +00:00
#[cfg(test)]
2020-07-31 16:30:37 +00:00
mod tests {
use test_utils::mark;
2020-05-06 08:16:55 +00:00
use crate::tests::{check_assist, check_assist_not_applicable, check_assist_target};
2019-09-20 15:55:59 +00:00
use super::*;
2019-09-20 15:55:59 +00:00
#[test]
fn make_raw_string_target() {
check_assist_target(
make_raw_string,
r#"
fn f() {
let s = <|>"random\nstring";
2019-09-20 15:55:59 +00:00
}
"#,
r#""random\nstring""#,
2019-09-20 15:55:59 +00:00
);
}
#[test]
fn make_raw_string_works() {
check_assist(
make_raw_string,
r#"
2020-06-23 22:30:34 +00:00
fn f() {
let s = <|>"random\nstring";
}
"#,
r##"
2020-06-23 22:30:34 +00:00
fn f() {
let s = r#"random
string"#;
2020-06-23 22:30:34 +00:00
}
"##,
2019-09-20 15:55:59 +00:00
)
}
2019-10-27 08:32:39 +00:00
#[test]
fn make_raw_string_works_inside_macros() {
check_assist(
make_raw_string,
r#"
fn f() {
format!(<|>"x = {}", 92)
}
"#,
r##"
fn f() {
format!(r#"x = {}"#, 92)
2019-10-27 08:32:39 +00:00
}
"##,
)
}
2019-09-20 15:55:59 +00:00
#[test]
fn make_raw_string_hashes_inside_works() {
2019-09-20 15:55:59 +00:00
check_assist(
make_raw_string,
r###"
2020-06-23 22:30:34 +00:00
fn f() {
let s = <|>"#random##\nstring";
}
"###,
r####"
2020-06-23 22:30:34 +00:00
fn f() {
let s = r#"#random##
string"#;
2020-06-23 22:30:34 +00:00
}
"####,
2019-09-20 15:55:59 +00:00
)
}
#[test]
fn make_raw_string_closing_hashes_inside_works() {
check_assist(
make_raw_string,
r###"
2020-06-23 22:30:34 +00:00
fn f() {
let s = <|>"#random\"##\nstring";
}
"###,
r####"
2020-06-23 22:30:34 +00:00
fn f() {
let s = r###"#random"##
string"###;
2020-06-23 22:30:34 +00:00
}
"####,
)
}
2019-09-20 15:55:59 +00:00
#[test]
fn make_raw_string_nothing_to_unescape_works() {
check_assist(
2019-09-20 15:55:59 +00:00
make_raw_string,
r#"
fn f() {
let s = <|>"random string";
2019-09-20 15:55:59 +00:00
}
"#,
r##"
fn f() {
let s = r#"random string"#;
}
"##,
)
2019-09-20 15:55:59 +00:00
}
#[test]
fn make_raw_string_not_works_on_partial_string() {
check_assist_not_applicable(
make_raw_string,
r#"
fn f() {
let s = "foo<|>
}
"#,
)
}
#[test]
fn make_usual_string_not_works_on_partial_string() {
check_assist_not_applicable(
make_usual_string,
r#"
fn main() {
let s = r#"bar<|>
}
"#,
)
}
2019-09-20 15:55:59 +00:00
#[test]
fn add_hash_target() {
check_assist_target(
add_hash,
r#"
fn f() {
let s = <|>r"random string";
}
"#,
r#"r"random string""#,
);
}
#[test]
fn add_hash_works() {
check_assist(
add_hash,
r#"
fn f() {
let s = <|>r"random string";
}
"#,
r##"
fn f() {
let s = r#"random string"#;
2019-09-20 15:55:59 +00:00
}
"##,
)
}
#[test]
fn add_more_hash_works() {
check_assist(
add_hash,
r##"
fn f() {
let s = <|>r#"random"string"#;
}
"##,
r###"
fn f() {
let s = r##"random"string"##;
2019-09-20 15:55:59 +00:00
}
"###,
)
}
#[test]
fn add_hash_not_works() {
check_assist_not_applicable(
add_hash,
r#"
fn f() {
let s = <|>"random string";
}
"#,
);
}
#[test]
fn remove_hash_target() {
check_assist_target(
remove_hash,
r##"
fn f() {
let s = <|>r#"random string"#;
}
"##,
r##"r#"random string"#"##,
);
}
#[test]
fn remove_hash_works() {
check_assist(
remove_hash,
r##"fn f() { let s = <|>r#"random string"#; }"##,
r#"fn f() { let s = r"random string"; }"#,
2019-09-20 15:55:59 +00:00
)
}
#[test]
fn cant_remove_required_hash() {
mark::check!(cant_remove_required_hash);
check_assist_not_applicable(
2019-09-20 15:55:59 +00:00
remove_hash,
r##"
fn f() {
let s = <|>r#"random"str"ing"#;
}
"##,
)
}
#[test]
fn remove_more_hash_works() {
check_assist(
remove_hash,
r###"
fn f() {
let s = <|>r##"random string"##;
}
"###,
r##"
fn f() {
let s = r#"random string"#;
2019-09-20 15:55:59 +00:00
}
"##,
)
}
#[test]
fn remove_hash_doesnt_work() {
check_assist_not_applicable(remove_hash, r#"fn f() { let s = <|>"random string"; }"#);
2019-09-20 15:55:59 +00:00
}
#[test]
fn remove_hash_no_hash_doesnt_work() {
check_assist_not_applicable(remove_hash, r#"fn f() { let s = <|>r"random string"; }"#);
2019-09-20 15:55:59 +00:00
}
#[test]
fn make_usual_string_target() {
check_assist_target(
make_usual_string,
r##"
fn f() {
let s = <|>r#"random string"#;
}
"##,
r##"r#"random string"#"##,
);
}
#[test]
fn make_usual_string_works() {
check_assist(
make_usual_string,
r##"
fn f() {
let s = <|>r#"random string"#;
}
"##,
r#"
fn f() {
let s = "random string";
2019-09-20 15:55:59 +00:00
}
"#,
)
}
#[test]
fn make_usual_string_with_quote_works() {
check_assist(
make_usual_string,
r##"
fn f() {
let s = <|>r#"random"str"ing"#;
}
"##,
r#"
fn f() {
let s = "random\"str\"ing";
2019-09-20 15:55:59 +00:00
}
"#,
)
}
#[test]
fn make_usual_string_more_hash_works() {
check_assist(
make_usual_string,
r###"
fn f() {
let s = <|>r##"random string"##;
}
"###,
r##"
fn f() {
let s = "random string";
2019-09-20 15:55:59 +00:00
}
"##,
)
}
#[test]
fn make_usual_string_not_works() {
check_assist_not_applicable(
make_usual_string,
r#"
fn f() {
let s = <|>"random string";
}
"#,
);
}
}