aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorFederico Igne <git@federicoigne.com>2021-11-05 22:58:26 +0000
committerFederico Igne <git@federicoigne.com>2021-11-05 23:00:47 +0000
commitc689e2ad8273b22e1ce8ae4ec58013d3e43010dc (patch)
tree83f27a3a0aed3728506da4af301b29ca795ddf68
parenta43a779ad12b464175f95d9381499fe28510ec09 (diff)
downloadexercism-c689e2ad8273b22e1ce8ae4ec58013d3e43010dc.tar.gz
exercism-c689e2ad8273b22e1ce8ae4ec58013d3e43010dc.zip
[rust] Anagram
-rw-r--r--rust/anagram/.gitignore8
-rw-r--r--rust/anagram/Cargo.toml4
-rw-r--r--rust/anagram/HELP.md85
-rw-r--r--rust/anagram/README.md61
-rw-r--r--rust/anagram/src/lib.rs30
-rw-r--r--rust/anagram/tests/anagram.rs173
6 files changed, 361 insertions, 0 deletions
diff --git a/rust/anagram/.gitignore b/rust/anagram/.gitignore
new file mode 100644
index 0000000..db7f315
--- /dev/null
+++ b/rust/anagram/.gitignore
@@ -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
8Cargo.lock
diff --git a/rust/anagram/Cargo.toml b/rust/anagram/Cargo.toml
new file mode 100644
index 0000000..5e06e4e
--- /dev/null
+++ b/rust/anagram/Cargo.toml
@@ -0,0 +1,4 @@
1[package]
2edition = "2021"
3name = "anagram"
4version = "0.0.0"
diff --git a/rust/anagram/HELP.md b/rust/anagram/HELP.md
new file mode 100644
index 0000000..b4252f8
--- /dev/null
+++ b/rust/anagram/HELP.md
@@ -0,0 +1,85 @@
1# Help
2
3## Running the tests
4
5Execute the tests with:
6
7```bash
8$ cargo test
9```
10
11All but the first test have been ignored. After you get the first test to
12pass, open the tests source file which is located in the `tests` directory
13and remove the `#[ignore]` flag from the next test and get the tests to pass
14again. Each separate test is a function with `#[test]` flag above it.
15Continue, until you pass every test.
16
17If you wish to run _only ignored_ tests without editing the tests source file, use:
18
19```bash
20$ cargo test -- --ignored
21```
22
23If you are using Rust 1.51 or later, you can run _all_ tests with
24
25```bash
26$ cargo test -- --include-ignored
27```
28
29To run a specific test, for example `some_test`, you can use:
30
31```bash
32$ cargo test some_test
33```
34
35If the specific test is ignored, use:
36
37```bash
38$ cargo test some_test -- --ignored
39```
40
41To learn more about Rust tests refer to the online [test documentation][rust-tests].
42
43[rust-tests]: https://doc.rust-lang.org/book/ch11-02-running-tests.html
44
45## Submitting your solution
46
47You can submit your solution using the `exercism submit src/lib.rs` command.
48This command will upload your solution to the Exercism website and print the solution page's URL.
49
50It's possible to submit an incomplete solution which allows you to:
51
52- See how others have completed the exercise
53- Request help from a mentor
54
55## Need to get help?
56
57If you'd like help solving the exercise, check the following pages:
58
59- The [Rust track's documentation](https://exercism.org/docs/tracks/rust)
60- [Exercism's support channel on gitter](https://gitter.im/exercism/support)
61- The [Frequently Asked Questions](https://exercism.org/docs/using/faqs)
62
63Should those resources not suffice, you could submit your (incomplete) solution to request mentoring.
64
65## Rust Installation
66
67Refer to the [exercism help page][help-page] for Rust installation and learning
68resources.
69
70## Submitting the solution
71
72Generally 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.
73
74## Feedback, Issues, Pull Requests
75
76The GitHub [track repository][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!
77
78If you want to know more about Exercism, take a look at the [contribution guide].
79
80## Submitting Incomplete Solutions
81It's possible to submit an incomplete solution so you can see how others have completed the exercise.
82
83[help-page]: https://exercism.io/tracks/rust/learning
84[github]: https://github.com/exercism/rust
85[contribution guide]: https://exercism.io/docs/community/contributors \ No newline at end of file
diff --git a/rust/anagram/README.md b/rust/anagram/README.md
new file mode 100644
index 0000000..e6be820
--- /dev/null
+++ b/rust/anagram/README.md
@@ -0,0 +1,61 @@
1# Anagram
2
3Welcome to Anagram on Exercism's Rust Track.
4If you need help running the tests or submitting your code, check out `HELP.md`.
5
6## Instructions
7
8An anagram is a rearrangement of letters to form a new word.
9Given a word and a list of candidates, select the sublist of anagrams of the given word.
10
11Given `"listen"` and a list of candidates like `"enlists" "google"
12"inlets" "banana"` the program should return a list containing
13`"inlets"`.
14
15The 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.
16
17The 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"]`.
18
19You 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.
20
21Try to limit case changes. Case changes are expensive in terms of time, so it's faster to minimize them.
22
23If sorting, consider [sort_unstable](https://doc.rust-lang.org/std/primitive.slice.html#method.sort_unstable) which is typically faster than stable sorting. When applicable, unstable sorting is preferred because it is generally faster than stable sorting and it doesn't allocate auxiliary memory.
24
25## Source
26
27### Created by
28
29- @EduardoBautista
30
31### Contributed to by
32
33- @andrewclarkson
34- @ashleygwilliams
35- @bobahop
36- @chancancode
37- @ClashTheBunny
38- @coriolinus
39- @cwhakes
40- @Dimkar3000
41- @EduardoBautista
42- @efx
43- @ErikSchierboom
44- @gris
45- @IanWhitney
46- @kytrinyx
47- @lutostag
48- @mkantor
49- @nfiles
50- @petertseng
51- @pminten
52- @quartsize
53- @rofrol
54- @stevejb71
55- @stringparser
56- @xakon
57- @ZapAnton
58
59### Based on
60
61Inspired by the Extreme Startup game - https://github.com/rchatley/extreme_startup \ No newline at end of file
diff --git a/rust/anagram/src/lib.rs b/rust/anagram/src/lib.rs
new file mode 100644
index 0000000..4602db0
--- /dev/null
+++ b/rust/anagram/src/lib.rs
@@ -0,0 +1,30 @@
1use std::collections::HashSet;
2
3pub fn anagrams_for<'a>(word: &str, possible_anagrams: &[&'a str]) -> HashSet<&'a str> {
4 let word = word.to_lowercase();
5 let word_sign = signature(&word);
6 possible_anagrams.iter().filter(|a| {
7 let a_lower = a.to_lowercase();
8 word_sign == signature(&a_lower) && word != a_lower
9 }).cloned().collect()
10}
11
12/// Compute the signature of a word.
13///
14/// In this case the signature of a word is the ordered list of its lowercased
15/// UTF8 characters.
16fn signature(word: &str) -> Vec<&str> {
17 let mut chars: Vec<&str> = vec![];
18 if !word.is_empty() {
19 let mut start = 0;
20 for i in (start+1)..word.len() {
21 if word.is_char_boundary(i) {
22 chars.push(word.get(start..i).unwrap());
23 start = i;
24 }
25 }
26 chars.push(word.get(start..).unwrap());
27 chars.sort_unstable();
28 }
29 chars
30}
diff --git a/rust/anagram/tests/anagram.rs b/rust/anagram/tests/anagram.rs
new file mode 100644
index 0000000..40fc03e
--- /dev/null
+++ b/rust/anagram/tests/anagram.rs
@@ -0,0 +1,173 @@
1use std::collections::HashSet;
2
3fn process_anagram_case(word: &str, inputs: &[&str], expected: &[&str]) {
4 let result = anagram::anagrams_for(word, inputs);
5
6 let expected: HashSet<&str> = expected.iter().cloned().collect();
7
8 assert_eq!(result, expected);
9}
10
11#[test]
12fn test_no_matches() {
13 let word = "diaper";
14
15 let inputs = ["hello", "world", "zombies", "pants"];
16
17 let outputs = vec![];
18
19 process_anagram_case(word, &inputs, &outputs);
20}
21
22#[test]
23fn test_detect_simple_anagram() {
24 let word = "ant";
25
26 let inputs = ["tan", "stand", "at"];
27
28 let outputs = vec!["tan"];
29
30 process_anagram_case(word, &inputs, &outputs);
31}
32
33#[test]
34fn test_does_not_confuse_different_duplicates() {
35 let word = "galea";
36
37 let inputs = ["eagle"];
38
39 let outputs = vec![];
40
41 process_anagram_case(word, &inputs, &outputs);
42}
43
44#[test]
45fn test_eliminate_anagram_subsets() {
46 let word = "good";
47
48 let inputs = ["dog", "goody"];
49
50 let outputs = vec![];
51
52 process_anagram_case(word, &inputs, &outputs);
53}
54
55#[test]
56fn test_detect_anagram() {
57 let word = "listen";
58
59 let inputs = ["enlists", "google", "inlets", "banana"];
60
61 let outputs = vec!["inlets"];
62
63 process_anagram_case(word, &inputs, &outputs);
64}
65
66#[test]
67fn test_multiple_anagrams() {
68 let word = "allergy";
69
70 let inputs = [
71 "gallery",
72 "ballerina",
73 "regally",
74 "clergy",
75 "largely",
76 "leading",
77 ];
78
79 let outputs = vec!["gallery", "regally", "largely"];
80
81 process_anagram_case(word, &inputs, &outputs);
82}
83
84#[test]
85fn test_case_insensitive_anagrams() {
86 let word = "Orchestra";
87
88 let inputs = ["cashregister", "Carthorse", "radishes"];
89
90 let outputs = vec!["Carthorse"];
91
92 process_anagram_case(word, &inputs, &outputs);
93}
94
95#[test]
96fn test_unicode_anagrams() {
97 let word = "ΑΒΓ";
98
99 // These words don't make sense, they're just greek letters cobbled together.
100 let inputs = ["ΒΓΑ", "ΒΓΔ", "γβα"];
101
102 let outputs = vec!["ΒΓΑ", "γβα"];
103
104 process_anagram_case(word, &inputs, &outputs);
105}
106
107#[test]
108fn test_misleading_unicode_anagrams() {
109 // Despite what a human might think these words contain different letters, the input uses Greek
110 // A and B while the list of potential anagrams uses Latin A and B.
111 let word = "ΑΒΓ";
112
113 let inputs = ["ABΓ"];
114
115 let outputs = vec![];
116
117 process_anagram_case(word, &inputs, &outputs);
118}
119
120#[test]
121fn test_does_not_detect_a_word_as_its_own_anagram() {
122 let word = "banana";
123
124 let inputs = ["banana"];
125
126 let outputs = vec![];
127
128 process_anagram_case(word, &inputs, &outputs);
129}
130
131#[test]
132fn test_does_not_detect_a_differently_cased_word_as_its_own_anagram() {
133 let word = "banana";
134
135 let inputs = ["bAnana"];
136
137 let outputs = vec![];
138
139 process_anagram_case(word, &inputs, &outputs);
140}
141
142#[test]
143fn test_does_not_detect_a_differently_cased_unicode_word_as_its_own_anagram() {
144 let word = "ΑΒΓ";
145
146 let inputs = ["ΑΒγ"];
147
148 let outputs = vec![];
149
150 process_anagram_case(word, &inputs, &outputs);
151}
152
153#[test]
154fn test_same_bytes_different_chars() {
155 let word = "a⬂"; // 61 E2 AC 82
156
157 let inputs = ["€a"]; // E2 82 AC 61
158
159 let outputs = vec![];
160
161 process_anagram_case(word, &inputs, &outputs);
162}
163
164#[test]
165fn test_different_words_but_same_ascii_sum() {
166 let word = "bc";
167
168 let inputs = ["ad"];
169
170 let outputs = vec![];
171
172 process_anagram_case(word, &inputs, &outputs);
173}