1 //! [![github]](https://github.com/dtolnay/remain) [![crates-io]](https://crates.io/crates/remain) [![docs-rs]](https://docs.rs/remain)
2 //!
3 //! [github]: https://img.shields.io/badge/github-8da0cb?style=for-the-badge&labelColor=555555&logo=github
4 //! [crates-io]: https://img.shields.io/badge/crates.io-fc8d62?style=for-the-badge&labelColor=555555&logo=rust
5 //! [docs-rs]: https://img.shields.io/badge/docs.rs-66c2a5?style=for-the-badge&labelColor=555555&logoColor=white&logo=
6 //!
7 //! <br>
8 //!
9 //! This crate provides an attribute macro to check at compile time that the
10 //! variants of an enum or the arms of a match expression are written in sorted
11 //! order.
12 //!
13 //! # Syntax
14 //!
15 //! Place a `#[remain::sorted]` attribute on enums, structs, match-expressions,
16 //! or let-statements whose value is a match-expression.
17 //!
18 //! Alternatively, import as `use remain::sorted;` and use `#[sorted]` as the
19 //! attribute.
20 //!
21 //! ```
22 //! # use std::error::Error as StdError;
23 //! # use std::fmt::{self, Display};
24 //! # use std::io;
25 //! #
26 //! #[remain::sorted]
27 //! #[derive(Debug)]
28 //! pub enum Error {
29 //! BlockSignal(signal::Error),
30 //! CreateCrasClient(libcras::Error),
31 //! CreateEventFd(sys_util::Error),
32 //! CreateSignalFd(sys_util::SignalFdError),
33 //! CreateSocket(io::Error),
34 //! DetectImageType(qcow::Error),
35 //! DeviceJail(io_jail::Error),
36 //! NetDeviceNew(virtio::NetError),
37 //! SpawnVcpu(io::Error),
38 //! }
39 //!
40 //! impl Display for Error {
41 //! # #[remain::check]
42 //! fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
43 //! use self::Error::*;
44 //!
45 //! #[remain::sorted]
46 //! match self {
47 //! BlockSignal(e) => write!(f, "failed to block signal: {}", e),
48 //! CreateCrasClient(e) => write!(f, "failed to create cras client: {}", e),
49 //! CreateEventFd(e) => write!(f, "failed to create eventfd: {}", e),
50 //! CreateSignalFd(e) => write!(f, "failed to create signalfd: {}", e),
51 //! CreateSocket(e) => write!(f, "failed to create socket: {}", e),
52 //! DetectImageType(e) => write!(f, "failed to detect disk image type: {}", e),
53 //! DeviceJail(e) => write!(f, "failed to jail device: {}", e),
54 //! NetDeviceNew(e) => write!(f, "failed to set up virtio networking: {}", e),
55 //! SpawnVcpu(e) => write!(f, "failed to spawn VCPU thread: {}", e),
56 //! }
57 //! }
58 //! }
59 //! #
60 //! # mod signal {
61 //! # pub use std::io::Error;
62 //! # }
63 //! #
64 //! # mod libcras {
65 //! # pub use std::io::Error;
66 //! # }
67 //! #
68 //! # mod sys_util {
69 //! # pub use std::io::{Error, Error as SignalFdError};
70 //! # }
71 //! #
72 //! # mod qcow {
73 //! # pub use std::io::Error;
74 //! # }
75 //! #
76 //! # mod io_jail {
77 //! # pub use std::io::Error;
78 //! # }
79 //! #
80 //! # mod virtio {
81 //! # pub use std::io::Error as NetError;
82 //! # }
83 //! #
84 //! # fn main() {}
85 //! ```
86 //!
87 //! If an enum variant, struct field, or match arm is inserted out of order,\
88 //!
89 //! ```diff
90 //! NetDeviceNew(virtio::NetError),
91 //! SpawnVcpu(io::Error),
92 //! + AaaUhOh(Box<dyn StdError>),
93 //! }
94 //! ```
95 //!
96 //! then the macro produces a compile error.
97 //!
98 //! ```console
99 //! error: AaaUhOh should sort before BlockSignal
100 //! --> tests/stable.rs:49:5
101 //! |
102 //! 49 | AaaUhOh(Box<dyn StdError>),
103 //! | ^^^^^^^
104 //! ```
105 //!
106 //! # Compiler support
107 //!
108 //! The attribute on enums is supported on any rustc version 1.31+.
109 //!
110 //! Rust does not yet have stable support for user-defined attributes within a
111 //! function body, so the attribute on match-expressions and let-statements
112 //! requires a nightly compiler and the following two features enabled:
113 //!
114 //! ```
115 //! # const IGNORE: &str = stringify! {
116 //! #![feature(proc_macro_hygiene, stmt_expr_attributes)]
117 //! # };
118 //! ```
119 //!
120 //! As a stable alternative, this crate provides a function-level attribute
121 //! called `#[remain::check]` which makes match-expression and let-statement
122 //! attributes work on any rustc version 1.31+. Place this attribute on any
123 //! function containing `#[sorted]` to make them work on a stable compiler.
124 //!
125 //! ```
126 //! # use std::fmt::{self, Display};
127 //! #
128 //! # enum Error {}
129 //! #
130 //! impl Display for Error {
131 //! #[remain::check]
132 //! fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
133 //! use self::Error::*;
134 //!
135 //! #[sorted]
136 //! match self {
137 //! /* ... */
138 //! # _ => unimplemented!(),
139 //! }
140 //! }
141 //! }
142 //! #
143 //! # fn main() {}
144 //! ```
145
146 #![allow(clippy::needless_doctest_main)]
147
148 extern crate proc_macro;
149
150 mod atom;
151 mod check;
152 mod compare;
153 mod emit;
154 mod format;
155 mod parse;
156 mod visit;
157
158 use proc_macro::TokenStream;
159 use quote::quote;
160 use syn::{parse_macro_input, ItemFn};
161
162 use crate::emit::emit;
163 use crate::parse::{Input, Nothing};
164
165 #[proc_macro_attribute]
sorted(args: TokenStream, input: TokenStream) -> TokenStream166 pub fn sorted(args: TokenStream, input: TokenStream) -> TokenStream {
167 let _ = parse_macro_input!(args as Nothing);
168 let mut input = parse_macro_input!(input as Input);
169 let kind = input.kind();
170
171 let result = check::sorted(&mut input);
172 let output = TokenStream::from(quote!(#input));
173
174 match result {
175 Ok(_) => output,
176 Err(err) => emit(err, kind, output),
177 }
178 }
179
180 #[proc_macro_attribute]
check(args: TokenStream, input: TokenStream) -> TokenStream181 pub fn check(args: TokenStream, input: TokenStream) -> TokenStream {
182 let _ = parse_macro_input!(args as Nothing);
183 let mut input = parse_macro_input!(input as ItemFn);
184
185 visit::check(&mut input);
186
187 TokenStream::from(quote!(#input))
188 }
189