aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorFederico Igne <git@federicoigne.com>2021-11-08 18:06:56 +0000
committerFederico Igne <git@federicoigne.com>2021-11-08 18:06:56 +0000
commitcd067503a7546eaee904a9b5c6ae30eb0175d376 (patch)
tree2f625fc40314fa2abbe60da496b6f0993af8a779
parentb6d7705471f0a583f1d115472ddbc8c4f8a420a9 (diff)
downloadexercism-cd067503a7546eaee904a9b5c6ae30eb0175d376.tar.gz
exercism-cd067503a7546eaee904a9b5c6ae30eb0175d376.zip
[rust] Macros
-rw-r--r--rust/macros/.gitignore8
-rw-r--r--rust/macros/Cargo.toml6
-rw-r--r--rust/macros/HELP.md85
-rw-r--r--rust/macros/README.md66
-rw-r--r--rust/macros/src/lib.rs12
-rw-r--r--rust/macros/tests/invalid/Cargo.toml55
-rw-r--r--rust/macros/tests/invalid/comma-sep.rs7
-rw-r--r--rust/macros/tests/invalid/double-commas.rs7
-rw-r--r--rust/macros/tests/invalid/leading-comma.rs7
-rw-r--r--rust/macros/tests/invalid/missing-argument.rs7
-rw-r--r--rust/macros/tests/invalid/no-comma.rs7
-rw-r--r--rust/macros/tests/invalid/only-arrow.rs7
-rw-r--r--rust/macros/tests/invalid/only-comma.rs7
-rw-r--r--rust/macros/tests/invalid/single-argument.rs7
-rw-r--r--rust/macros/tests/invalid/triple-arguments.rs7
-rw-r--r--rust/macros/tests/invalid/two-arrows.rs7
-rw-r--r--rust/macros/tests/macros.rs200
17 files changed, 502 insertions, 0 deletions
diff --git a/rust/macros/.gitignore b/rust/macros/.gitignore
new file mode 100644
index 0000000..db7f315
--- /dev/null
+++ b/rust/macros/.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/macros/Cargo.toml b/rust/macros/Cargo.toml
new file mode 100644
index 0000000..b7d10d7
--- /dev/null
+++ b/rust/macros/Cargo.toml
@@ -0,0 +1,6 @@
1[package]
2edition = "2018"
3name = "macros"
4version = "0.1.0"
5
6[dependencies]
diff --git a/rust/macros/HELP.md b/rust/macros/HELP.md
new file mode 100644
index 0000000..b4252f8
--- /dev/null
+++ b/rust/macros/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/macros/README.md b/rust/macros/README.md
new file mode 100644
index 0000000..c1e6015
--- /dev/null
+++ b/rust/macros/README.md
@@ -0,0 +1,66 @@
1# Macros
2
3Welcome to Macros on Exercism's Rust Track.
4If you need help running the tests or submitting your code, check out `HELP.md`.
5
6## Instructions
7
8Macros are a powerful part of a Rust programmer's toolkit, and [macros by example](https://doc.rust-lang.org/reference/macros-by-example.html) are a relatively simple way to access this power. Let's write one!
9
10## Context
11
12What is a macro? [Wikipedia](https://en.wikipedia.org/wiki/Macro_(computer_science)) describes it thus:
13
14> A macro (short for "macroinstruction", from Greek μακρός 'long') in computer science is a rule or pattern that specifies how a certain input sequence (often a sequence of characters) should be mapped to a replacement output sequence (also often a sequence of characters) according to a defined procedure. The mapping process that instantiates (transforms) a macro use into a specific sequence is known as macro expansion.
15
16Illuminating! But to be more concrete, macros are a special syntax which allows you to generate code at compile time. Macros can be used for compile-time calculation, but more often they're just another way to abstract your code. For example, you've probably already used `println!()` and `vec![]`. These each take an arbitrary number of arguments, so you can't express them as simple functions. On the other hand, they always expand to some amount of absolutely standard Rust code. If you're interested, you can use the [cargo expand](https://github.com/dtolnay/cargo-expand) subcommand to view the results of macro expansion in your code.
17
18For further information about macros in Rust, The Rust Book has a [good chapter](https://doc.rust-lang.org/book/ch19-06-macros.html) on them.
19
20## Problem Statement
21
22You can produce a `Vec` of arbitrary length inline by using the `vec![]` macro. However, Rust doesn't come with a way to produce a [`HashMap`](https://doc.rust-lang.org/std/collections/struct.HashMap.html) inline. Rectify this by writing a `hashmap!()` macro.
23
24For example, a user of your library might write `hashmap!('a' => 3, 'b' => 11, 'z' => 32)`. This should expand to the following code:
25
26```rust
27{
28 let mut hm = HashMap::new();
29 hm.insert('a', 3);
30 hm.insert('b', 11);
31 hm.insert('z', 32);
32 hm
33}
34```
35
36Note that the [`maplit` crate](https://crates.io/crates/maplit) provides a macro which perfectly solves this exercise. Please implement your own solution instead of using this crate; please make an attempt on your own before viewing its source.
37
38Note that this exercise requires Rust 1.36 or later.
39
40## Source
41
42### Created by
43
44- @coriolinus
45
46### Contributed to by
47
48- @bantic
49- @cwhakes
50- @DarthStrom
51- @efx
52- @Emerentius
53- @ErikSchierboom
54- @lutostag
55- @pedantic79
56- @petertseng
57- @rofrol
58- @ssomers
59- @stringparser
60- @tjade273
61- @xakon
62- @ZapAnton
63
64### Based on
65
66Peter Goodspeed-Niklaus \ No newline at end of file
diff --git a/rust/macros/src/lib.rs b/rust/macros/src/lib.rs
new file mode 100644
index 0000000..267aa46
--- /dev/null
+++ b/rust/macros/src/lib.rs
@@ -0,0 +1,12 @@
1
2#[macro_export]
3macro_rules! hashmap {
4 ( $($k:expr => $v:expr),+, ) => { $crate::hashmap!($($k => $v),+) };
5 ( $($k:expr => $v:expr),* ) => {
6 {
7 let mut hm = ::std::collections::HashMap::new();
8 $( hm.insert($k, $v); )*
9 hm
10 }
11 };
12}
diff --git a/rust/macros/tests/invalid/Cargo.toml b/rust/macros/tests/invalid/Cargo.toml
new file mode 100644
index 0000000..b096d99
--- /dev/null
+++ b/rust/macros/tests/invalid/Cargo.toml
@@ -0,0 +1,55 @@
1#
2# This Cargo.toml file is used by the simple-trybuild module.
3# When adding a new file, please name the [[bin]] name to match the file
4# it is used to produce an error message
5#
6
7[package]
8name = "macros-tests"
9version = "0.0.0"
10edition = "2018"
11publish = false
12
13[dependencies.macros]
14path = "../../"
15default-features = false
16
17[[bin]]
18name = "comma-sep-rs"
19path = "comma-sep.rs"
20
21[[bin]]
22name = "double-commas-rs"
23path = "double-commas.rs"
24
25[[bin]]
26name = "only-arrow-rs"
27path = "only-arrow.rs"
28
29[[bin]]
30name = "only-comma-rs"
31path = "only-comma.rs"
32
33[[bin]]
34name = "single-argument-rs"
35path = "single-argument.rs"
36
37[[bin]]
38name = "triple-arguments-rs"
39path = "triple-arguments.rs"
40
41[[bin]]
42name = "two-arrows-rs"
43path = "two-arrows.rs"
44
45[[bin]]
46name = "leading-comma-rs"
47path = "leading-comma.rs"
48
49[[bin]]
50name = "no-comma-rs"
51path = "no-comma.rs"
52
53[[bin]]
54name = "missing-argument-rs"
55path = "missing-argument.rs"
diff --git a/rust/macros/tests/invalid/comma-sep.rs b/rust/macros/tests/invalid/comma-sep.rs
new file mode 100644
index 0000000..c9b6f44
--- /dev/null
+++ b/rust/macros/tests/invalid/comma-sep.rs
@@ -0,0 +1,7 @@
1use macros::hashmap;
2use std::collections::HashMap;
3
4fn main() {
5 // using only commas is invalid
6 let _hm: HashMap<_, _> = hashmap!('a', 1);
7}
diff --git a/rust/macros/tests/invalid/double-commas.rs b/rust/macros/tests/invalid/double-commas.rs
new file mode 100644
index 0000000..d52dec6
--- /dev/null
+++ b/rust/macros/tests/invalid/double-commas.rs
@@ -0,0 +1,7 @@
1use macros::hashmap;
2use std::collections::HashMap;
3
4fn main() {
5 // a single trailing comma is okay, but two is not
6 let _hm: HashMap<_, _> = hashmap!('a' => 2, ,);
7}
diff --git a/rust/macros/tests/invalid/leading-comma.rs b/rust/macros/tests/invalid/leading-comma.rs
new file mode 100644
index 0000000..bc03666
--- /dev/null
+++ b/rust/macros/tests/invalid/leading-comma.rs
@@ -0,0 +1,7 @@
1use macros::hashmap;
2use std::collections::HashMap;
3
4fn main() {
5 // leading commas are not valid
6 let _hm: HashMap<_, _> = hashmap!(, 'a' => 2);
7}
diff --git a/rust/macros/tests/invalid/missing-argument.rs b/rust/macros/tests/invalid/missing-argument.rs
new file mode 100644
index 0000000..893d609
--- /dev/null
+++ b/rust/macros/tests/invalid/missing-argument.rs
@@ -0,0 +1,7 @@
1use macros::hashmap;
2use std::collections::HashMap;
3
4fn main() {
5 // an argument should come between each pair of commas
6 let _hm: HashMap<_, _> = hashmap!('a' => 1, , 'b' => 2);
7}
diff --git a/rust/macros/tests/invalid/no-comma.rs b/rust/macros/tests/invalid/no-comma.rs
new file mode 100644
index 0000000..c4e6a92
--- /dev/null
+++ b/rust/macros/tests/invalid/no-comma.rs
@@ -0,0 +1,7 @@
1use macros::hashmap;
2use std::collections::HashMap;
3
4fn main() {
5 // Key value pairs must be separated by commas
6 let _hm: HashMap<_, _> = hashmap!('a' => 1 'b' => 2);
7}
diff --git a/rust/macros/tests/invalid/only-arrow.rs b/rust/macros/tests/invalid/only-arrow.rs
new file mode 100644
index 0000000..7f7d730
--- /dev/null
+++ b/rust/macros/tests/invalid/only-arrow.rs
@@ -0,0 +1,7 @@
1use macros::hashmap;
2use std::collections::HashMap;
3
4fn main() {
5 // a single random arrow is not valid
6 let _hm: HashMap<(), ()> = hashmap!(=>);
7}
diff --git a/rust/macros/tests/invalid/only-comma.rs b/rust/macros/tests/invalid/only-comma.rs
new file mode 100644
index 0000000..3b863b3
--- /dev/null
+++ b/rust/macros/tests/invalid/only-comma.rs
@@ -0,0 +1,7 @@
1use macros::hashmap;
2use std::collections::HashMap;
3
4fn main() {
5 // a single random comma is not valid
6 let _hm: HashMap<(), ()> = hashmap!(,);
7}
diff --git a/rust/macros/tests/invalid/single-argument.rs b/rust/macros/tests/invalid/single-argument.rs
new file mode 100644
index 0000000..f8ac092
--- /dev/null
+++ b/rust/macros/tests/invalid/single-argument.rs
@@ -0,0 +1,7 @@
1use macros::hashmap;
2use std::collections::HashMap;
3
4fn main() {
5 // a single argument is invalid
6 let _hm: HashMap<_, _> = hashmap!('a');
7}
diff --git a/rust/macros/tests/invalid/triple-arguments.rs b/rust/macros/tests/invalid/triple-arguments.rs
new file mode 100644
index 0000000..54351ae
--- /dev/null
+++ b/rust/macros/tests/invalid/triple-arguments.rs
@@ -0,0 +1,7 @@
1use macros::hashmap;
2use std::collections::HashMap;
3
4fn main() {
5 // three arguments are invalid
6 hashmap!('a' => 1, 'b');
7}
diff --git a/rust/macros/tests/invalid/two-arrows.rs b/rust/macros/tests/invalid/two-arrows.rs
new file mode 100644
index 0000000..254a82e
--- /dev/null
+++ b/rust/macros/tests/invalid/two-arrows.rs
@@ -0,0 +1,7 @@
1use macros::hashmap;
2use std::collections::HashMap;
3
4fn main() {
5 // a trailing => isn't valid either
6 hashmap!('a' => 2, =>);
7}
diff --git a/rust/macros/tests/macros.rs b/rust/macros/tests/macros.rs
new file mode 100644
index 0000000..077a9e7
--- /dev/null
+++ b/rust/macros/tests/macros.rs
@@ -0,0 +1,200 @@
1use macros::hashmap;
2use std::collections::HashMap;
3
4#[test]
5fn test_empty() {
6 let expected: HashMap<u32, u32> = HashMap::new();
7 let computed: HashMap<u32, u32> = hashmap!();
8 assert_eq!(computed, expected);
9}
10
11#[test]
12fn test_single() {
13 let mut expected = HashMap::new();
14 expected.insert(1, "one");
15 assert_eq!(hashmap!(1 => "one"), expected);
16}
17
18#[test]
19fn test_no_trailing_comma() {
20 let mut expected = HashMap::new();
21 expected.insert(1, "one");
22 expected.insert(2, "two");
23 assert_eq!(hashmap!(1 => "one", 2 => "two"), expected);
24}
25
26#[test]
27fn test_trailing_comma() {
28 let mut expected = HashMap::new();
29 expected.insert('h', 89);
30 expected.insert('a', 1);
31 expected.insert('s', 19);
32 expected.insert('h', 8);
33 assert_eq!(
34 hashmap!(
35 'h' => 89,
36 'a' => 1,
37 's' => 19,
38 'h' => 8,
39 ),
40 expected
41 );
42}
43
44#[test]
45fn test_nested() {
46 let mut expected = HashMap::new();
47 expected.insert("non-empty", {
48 let mut subhashmap = HashMap::new();
49 subhashmap.insert(23, 623);
50 subhashmap.insert(34, 21);
51 subhashmap
52 });
53 expected.insert("empty", HashMap::new());
54 assert_eq!(
55 hashmap!(
56 "non-empty" => hashmap!(
57 23 => 623,
58 34 => 21
59 ),
60 "empty" => hashmap!()
61 ),
62 expected
63 );
64}
65
66mod test {
67 #[test]
68 fn type_not_in_scope() {
69 use macros::hashmap;
70
71 let _empty: ::std::collections::HashMap<(), ()> = hashmap!();
72 let _without_comma = hashmap!(23=> 623, 34 => 21);
73 let _with_trailing = hashmap!(23 => 623, 34 => 21,);
74 }
75
76 #[test]
77 fn test_macro_out_of_scope() {
78 let _empty: ::std::collections::HashMap<(), ()> = macros::hashmap!();
79 let _without_comma = macros::hashmap!(23=> 623, 34 => 21);
80 let _with_trailing = macros::hashmap!(23 => 623, 34 => 21,);
81 }
82}
83
84#[test]
85fn test_type_override() {
86 // The macro should always use std::collections::HashMap and ignore crate::std::collections::HashMap
87 mod std {
88 pub mod collections {
89 pub struct HashMap;
90
91 impl HashMap {
92 #[allow(dead_code)]
93 pub fn new() -> Self {
94 panic!("Do not allow users to override which HashMap is used");
95 }
96
97 #[allow(dead_code)]
98 pub fn insert<K, V>(&mut self, _key: K, _val: V) {
99 panic!("Do not allow users to override which HashMap is used");
100 }
101 }
102 }
103 }
104
105 let _empty: ::std::collections::HashMap<(), ()> = hashmap!();
106 let _without_comma = hashmap!(1 => 2, 3 => 4);
107 let _with_trailing = hashmap!(1 => 2, 3 => 4,);
108}
109
110#[test]
111fn test_compile_fails_comma_sep() {
112 simple_trybuild::compile_fail("comma-sep.rs");
113}
114
115#[test]
116fn test_compile_fails_double_commas() {
117 simple_trybuild::compile_fail("double-commas.rs");
118}
119
120#[test]
121fn test_compile_fails_only_comma() {
122 simple_trybuild::compile_fail("only-comma.rs");
123}
124
125#[test]
126fn test_compile_fails_single_argument() {
127 simple_trybuild::compile_fail("single-argument.rs");
128}
129
130#[test]
131fn test_compile_fails_triple_arguments() {
132 simple_trybuild::compile_fail("triple-arguments.rs");
133}
134
135#[test]
136fn test_compile_fails_only_arrow() {
137 simple_trybuild::compile_fail("only-arrow.rs");
138}
139
140#[test]
141fn test_compile_fails_two_arrows() {
142 simple_trybuild::compile_fail("two-arrows.rs");
143}
144
145#[test]
146fn test_compile_fails_leading_comma() {
147 simple_trybuild::compile_fail("leading-comma.rs");
148}
149
150#[test]
151fn test_compile_fails_no_comma() {
152 simple_trybuild::compile_fail("no-comma.rs");
153}
154
155#[test]
156fn test_compile_fails_missing_argument() {
157 simple_trybuild::compile_fail("missing-argument.rs");
158}
159
160mod simple_trybuild {
161 use std::path::PathBuf;
162 use std::process::Command;
163
164 pub fn compile_fail(file_name: &str) {
165 let invalid_path: PathBuf = ["tests", "invalid"].iter().collect::<PathBuf>();
166
167 let mut file_path = invalid_path.clone();
168 file_path.push(file_name);
169 assert!(
170 file_path.exists(),
171 "{:?} does not exist.",
172 file_path.into_os_string()
173 );
174
175 let test_name = file_name.replace(".", "-");
176 let macros_dir = ["..", "..", "target", "tests", "macros"]
177 .iter()
178 .collect::<PathBuf>();
179
180 let result = Command::new("cargo")
181 .current_dir(invalid_path)
182 .arg("build")
183 .arg("--offline")
184 .arg("--target-dir")
185 .arg(macros_dir)
186 .arg("--bin")
187 .arg(test_name)
188 .output();
189
190 if let Ok(result) = result {
191 assert!(
192 !result.status.success(),
193 "Expected {:?} to fail to compile, but it succeeded.",
194 file_path
195 );
196 } else {
197 panic!("Running subprocess failed.");
198 }
199 }
200}