Skip to content

Commit 9cbc3aa

Browse files
[Rust] Anagram
1 parent e272a0d commit 9cbc3aa

File tree

6 files changed

+311
-0
lines changed

6 files changed

+311
-0
lines changed

rust/Cargo.lock

+22
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

rust/anagram/.gitignore

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
# Generated by Cargo
2+
# will have compiled files and executables
3+
/target/
4+
**/*.rs.bk
5+
6+
# Remove Cargo.lock from gitignore if creating an executable, leave it for libraries
7+
# More information here http://doc.crates.io/guide.html#cargotoml-vs-cargolock
8+
Cargo.lock

rust/anagram/Cargo.toml

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
[package]
2+
edition = "2018"
3+
name = "anagram"
4+
version = "0.0.0"
5+
6+
[dependencies]
7+
itertools = "0.9.0"

rust/anagram/README.md

+97
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,97 @@
1+
# Anagram
2+
3+
An anagram is a rearrangement of letters to form a new word.
4+
Given a word and a list of candidates, select the sublist of anagrams of the given word.
5+
6+
Given `"listen"` and a list of candidates like `"enlists" "google"
7+
"inlets" "banana"` the program should return a list containing
8+
`"inlets"`.
9+
10+
# Hints
11+
12+
The solution is case insensitive, which means `"WOrd"` is the same as `"word"` or `"woRd"`. It may help to take a peek at the [std library](https://doc.rust-lang.org/std/primitive.char.html) for functions that can convert between them.
13+
14+
The solution cannot contain the input word. A word is always an anagram of itself, which means it is not an interesting result. Given `"hello"` and the list `["hello", "olleh"]` the answer is `["olleh"]`.
15+
16+
You are going to have to adjust the function signature provided in the stub in order for the lifetimes to work out properly. This is intentional: what's there demonstrates the basics of lifetime syntax, and what's missing teaches how to interpret lifetime-related compiler errors.
17+
18+
19+
## Rust Installation
20+
21+
Refer to the [exercism help page][help-page] for Rust installation and learning
22+
resources.
23+
24+
## Writing the Code
25+
26+
Execute the tests with:
27+
28+
```bash
29+
$ cargo test
30+
```
31+
32+
All but the first test have been ignored. After you get the first test to
33+
pass, open the tests source file which is located in the `tests` directory
34+
and remove the `#[ignore]` flag from the next test and get the tests to pass
35+
again. Each separate test is a function with `#[test]` flag above it.
36+
Continue, until you pass every test.
37+
38+
If you wish to run all ignored tests without editing the tests source file, use:
39+
40+
```bash
41+
$ cargo test -- --ignored
42+
```
43+
44+
To run a specific test, for example `some_test`, you can use:
45+
46+
```bash
47+
$ cargo test some_test
48+
```
49+
50+
If the specific test is ignored use:
51+
52+
```bash
53+
$ cargo test some_test -- --ignored
54+
```
55+
56+
To learn more about Rust tests refer to the [online test documentation][rust-tests]
57+
58+
Make sure to read the [Modules][modules] chapter if you
59+
haven't already, it will help you with organizing your files.
60+
61+
## Further improvements
62+
63+
After you have solved the exercise, please consider using the additional utilities, described in the [installation guide](https://exercism.io/tracks/rust/installation), to further refine your final solution.
64+
65+
To format your solution, inside the solution directory use
66+
67+
```bash
68+
cargo fmt
69+
```
70+
71+
To see, if your solution contains some common ineffective use cases, inside the solution directory use
72+
73+
```bash
74+
cargo clippy --all-targets
75+
```
76+
77+
## Submitting the solution
78+
79+
Generally you should submit all files in which you implemented your solution (`src/lib.rs` in most cases). If you are using any external crates, please consider submitting the `Cargo.toml` file. This will make the review process faster and clearer.
80+
81+
## Feedback, Issues, Pull Requests
82+
83+
The [exercism/rust](https://github.com/exercism/rust) repository on GitHub is the home for all of the Rust exercises. If you have feedback about an exercise, or want to help implement new exercises, head over there and create an issue. Members of the rust track team are happy to help!
84+
85+
If you want to know more about Exercism, take a look at the [contribution guide](https://github.com/exercism/docs/blob/master/contributing-to-language-tracks/README.md).
86+
87+
[help-page]: https://exercism.io/tracks/rust/learning
88+
[modules]: https://doc.rust-lang.org/book/ch07-02-defining-modules-to-control-scope-and-privacy.html
89+
[cargo]: https://doc.rust-lang.org/book/ch14-00-more-about-cargo.html
90+
[rust-tests]: https://doc.rust-lang.org/book/ch11-02-running-tests.html
91+
92+
## Source
93+
94+
Inspired by the Extreme Startup game [https://github.com/rchatley/extreme_startup](https://github.com/rchatley/extreme_startup)
95+
96+
## Submitting Incomplete Solutions
97+
It's possible to submit an incomplete solution so you can see how others have completed the exercise.

rust/anagram/src/lib.rs

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
use std::collections::HashSet;
2+
use itertools::Itertools;
3+
4+
fn is_anagram(a: String, b: String) -> bool {
5+
a != b && a.chars().sorted().collect::<String>() == b.chars().sorted().collect::<String>()
6+
}
7+
8+
pub fn anagrams_for<'a>(word: &str, possible_anagrams: &'a [&str]) -> HashSet<&'a str> {
9+
possible_anagrams
10+
.iter()
11+
.copied()
12+
.filter(|&w| is_anagram(w.to_lowercase(), word.to_lowercase()))
13+
.collect()
14+
}

rust/anagram/tests/anagram.rs

+163
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,163 @@
1+
use std::collections::HashSet;
2+
use std::iter::FromIterator;
3+
4+
fn process_anagram_case(word: &str, inputs: &[&str], expected: &[&str]) {
5+
let result = anagram::anagrams_for(word, inputs);
6+
7+
let expected: HashSet<&str> = HashSet::from_iter(expected.iter().cloned());
8+
9+
assert_eq!(result, expected);
10+
}
11+
12+
#[test]
13+
fn test_no_matches() {
14+
let word = "diaper";
15+
16+
let inputs = ["hello", "world", "zombies", "pants"];
17+
18+
let outputs = vec![];
19+
20+
process_anagram_case(word, &inputs, &outputs);
21+
}
22+
23+
#[test]
24+
fn test_detect_simple_anagram() {
25+
let word = "ant";
26+
27+
let inputs = ["tan", "stand", "at"];
28+
29+
let outputs = vec!["tan"];
30+
31+
process_anagram_case(word, &inputs, &outputs);
32+
}
33+
34+
#[test]
35+
fn test_does_not_confuse_different_duplicates() {
36+
let word = "galea";
37+
38+
let inputs = ["eagle"];
39+
40+
let outputs = vec![];
41+
42+
process_anagram_case(word, &inputs, &outputs);
43+
}
44+
45+
#[test]
46+
fn test_eliminate_anagram_subsets() {
47+
let word = "good";
48+
49+
let inputs = ["dog", "goody"];
50+
51+
let outputs = vec![];
52+
53+
process_anagram_case(word, &inputs, &outputs);
54+
}
55+
56+
#[test]
57+
fn test_detect_anagram() {
58+
let word = "listen";
59+
60+
let inputs = ["enlists", "google", "inlets", "banana"];
61+
62+
let outputs = vec!["inlets"];
63+
64+
process_anagram_case(word, &inputs, &outputs);
65+
}
66+
67+
#[test]
68+
fn test_multiple_anagrams() {
69+
let word = "allergy";
70+
71+
let inputs = [
72+
"gallery",
73+
"ballerina",
74+
"regally",
75+
"clergy",
76+
"largely",
77+
"leading",
78+
];
79+
80+
let outputs = vec!["gallery", "regally", "largely"];
81+
82+
process_anagram_case(word, &inputs, &outputs);
83+
}
84+
85+
#[test]
86+
fn test_case_insensitive_anagrams() {
87+
let word = "Orchestra";
88+
89+
let inputs = ["cashregister", "Carthorse", "radishes"];
90+
91+
let outputs = vec!["Carthorse"];
92+
93+
process_anagram_case(word, &inputs, &outputs);
94+
}
95+
96+
#[test]
97+
fn test_unicode_anagrams() {
98+
let word = "ΑΒΓ";
99+
100+
// These words don't make sense, they're just greek letters cobbled together.
101+
let inputs = ["ΒΓΑ", "ΒΓΔ", "γβα"];
102+
103+
let outputs = vec!["ΒΓΑ", "γβα"];
104+
105+
process_anagram_case(word, &inputs, &outputs);
106+
}
107+
108+
#[test]
109+
fn test_misleading_unicode_anagrams() {
110+
// Despite what a human might think these words different letters, the input uses Greek A and B
111+
// while the list of potential anagrams uses Latin A and B.
112+
let word = "ΑΒΓ";
113+
114+
let inputs = ["ABΓ"];
115+
116+
let outputs = vec![];
117+
118+
process_anagram_case(word, &inputs, &outputs);
119+
}
120+
121+
#[test]
122+
fn test_does_not_detect_a_word_as_its_own_anagram() {
123+
let word = "banana";
124+
125+
let inputs = ["banana"];
126+
127+
let outputs = vec![];
128+
129+
process_anagram_case(word, &inputs, &outputs);
130+
}
131+
132+
#[test]
133+
fn test_does_not_detect_a_differently_cased_word_as_its_own_anagram() {
134+
let word = "banana";
135+
136+
let inputs = ["bAnana"];
137+
138+
let outputs = vec![];
139+
140+
process_anagram_case(word, &inputs, &outputs);
141+
}
142+
143+
#[test]
144+
fn test_does_not_detect_a_differently_cased_unicode_word_as_its_own_anagram() {
145+
let word = "ΑΒΓ";
146+
147+
let inputs = ["ΑΒγ"];
148+
149+
let outputs = vec![];
150+
151+
process_anagram_case(word, &inputs, &outputs);
152+
}
153+
154+
#[test]
155+
fn test_same_bytes_different_chars() {
156+
let word = "a⬂"; // 61 E2 AC 82
157+
158+
let inputs = ["€a"]; // E2 82 AC 61
159+
160+
let outputs = vec![];
161+
162+
process_anagram_case(word, &inputs, &outputs);
163+
}

0 commit comments

Comments
 (0)