1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
pub fn camelcase_to_snakecase(s: &str) -> String {
    let mut words = vec![];
    let mut word = String::from("");

    for c in s.chars() {
        if c.is_uppercase() {
            // flush
            words.push(word);
            word = String::from("");
        }
        word.push(c);
    }

    words.push(word);

    words
        .into_iter()
        .filter(|w| !w.is_empty())
        .collect::<Vec<_>>()
        .join("_")
        .to_lowercase()
}

#[test]
fn test_camelcase_to_snakecase() {
    assert_eq!(camelcase_to_snakecase("FooBar"), "foo_bar");
    assert_eq!(camelcase_to_snakecase("Foo"), "foo");
    assert_eq!(camelcase_to_snakecase("F"), "f");
}

pub fn escape_rust_keyword(s: &str) -> String {
    match s {
        "const" | "as" | "else" | "self" | "break" | "false" | "for" | "if" | "return" | "str"
        | "super" | "true" | "while" | "yield" => format!("{}_", s),
        _ => s.to_string(),
    }
}

#[test]
fn test_escape_rust_keyword() {
    assert_eq!(escape_rust_keyword("foo"), "foo");
    assert_eq!(escape_rust_keyword("while"), "while_");
}

pub fn snakecase_to_camelcase(s: &str) -> String {
    fn capitalize_word(s: &str) -> String {
        let mut c = s.chars();
        match c.next() {
            None => String::new(),
            Some(f) => f.to_uppercase().collect::<String>() + c.as_str(),
        }
    }

    s.split('_').map(capitalize_word).collect()
}

#[test]
fn test_snakecase_to_camelcase() {
    assert_eq!(snakecase_to_camelcase("foo_bar"), "FooBar");
    assert_eq!(snakecase_to_camelcase("foo"), "Foo");
    assert_eq!(snakecase_to_camelcase("f"), "F");
}

pub fn escape_c_keyword(s: &str) -> String {
    match s {
        "default" | "operator" | "else" | "const" | "and" | "break" | "case" | "class"
        | "false" | "float" | "for" | "if" | "int" | "or" | "return" | "true" | "while" => {
            format!("{}_", s)
        }
        _ => s.to_string(),
    }
}

#[test]
fn test_escape_c_keyword() {
    assert_eq!(escape_c_keyword("foo"), "foo");
    assert_eq!(escape_c_keyword("default"), "default_");
}

pub fn escape_cpp_keyword(s: &str) -> String {
    match s {
        "default" | "operator" | "else" | "const" | "and" | "break" | "case" | "class"
        | "false" | "float" | "for" | "if" | "int" | "or" | "return" | "true" | "while"
        | "ERANGE" => format!("{}_", s),
        _ => s.to_string(),
    }
}

#[test]
fn test_escape_cpp_keyword() {
    assert_eq!(escape_cpp_keyword("foo"), "foo");
    assert_eq!(escape_cpp_keyword("default"), "default_");
}

pub fn escape_js_keyword(s: &str) -> String {
    match s {
        "const" | "default" | "var" | "else" => format!("{}_", s),
        _ => s.to_string(),
    }
}

#[test]
fn test_escape_js_keyword() {
    assert_eq!(escape_js_keyword("var"), "var_");
    assert_eq!(escape_js_keyword("foo"), "foo");
}