Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add try_setter option and generate appropriate methods #72

Merged
merged 17 commits into from
Apr 12, 2017
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion .travis.yml
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ matrix:
- rust: 1.15.0
env: JOB=build CARGO_FEATURES="struct_default"
- rust: nightly
env: JOB=build CARGO_FEATURES="compiletests logging"
env: JOB=build CARGO_FEATURES="nightlytests logging"
- rust: nightly
env: JOB=style_check
allow_failures:
Expand Down
10 changes: 10 additions & 0 deletions derive_builder/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,16 @@
All notable changes to this project will be documented in this file.
This project adheres to [Semantic Versioning](http://semver.org/).

## [0.4.3] - 2017-04-13

### Added
- try_setters, e.g. `#[builder(try_setter)]`. These setters are exposed alongside the
normal field setters and allow callers to pass in values which have fallible
conversions to the needed type through `TryInto`. This attribute can be used on any
Rust channel, but will only emit setters on nightly when `#![feature(try_from)]` is
declared in the consuming crate's root; this will change when Rust issue
[#33417](https://github.com/rust-lang/rust/issues/33417) is resolved.
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nice summary! 👍


## [0.4.2] - 2017-04-10

### Fixed
Expand Down
1 change: 0 additions & 1 deletion derive_builder/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,6 @@ proc-macro = true
logging = [ "log", "env_logger", "derive_builder_core/logging" ]
struct_default = []
skeptic_tests = ["skeptic"]
try_setter = []

[dependencies]
syn = "0.11"
Expand Down
52 changes: 52 additions & 0 deletions derive_builder/examples/try_setter.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
//! This example illustrates the use of try-setters.
#![feature(try_from)]

#[macro_use]
extern crate derive_builder;

use std::convert::{From, TryFrom};
use std::net::{IpAddr, AddrParseError};
use std::str::FromStr;
use std::string::ToString;

/// Temporary newtype hack around lack of TryFrom implementations
/// in std. The rust-lang issue on the subject says that there will be a
/// blanket impl for everything that currently implements FromStr, which
/// will make this feature much more useful for input validation.
#[derive(Debug, Clone, PartialEq)]
pub struct MyAddr(IpAddr);

impl From<IpAddr> for MyAddr {
fn from(v: IpAddr) -> Self {
MyAddr(v)
}
}

impl<'a> TryFrom<&'a str> for MyAddr {
type Err = AddrParseError;

fn try_from(v: &str) -> Result<Self, AddrParseError> {
Ok(MyAddr(IpAddr::from_str(v)?))
}
}

#[derive(Builder, Debug, PartialEq)]
#[builder(try_setter, setter(into))]
struct Lorem {
pub name: String,
pub addr: MyAddr,
}

fn main() {
create("Jane", "1.2.3.4").unwrap();
create("Bobby", "").unwrap_err();
}

fn create(name: &str, addr: &str) -> Result<Lorem, String> {
// Fallible and infallible setters can be mixed freely when using
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@colin-kiegel can you please try running cargo expand --example try_setter on this and watch what happens to the comment? Something weird is happening here, but I'm not sure what.

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Let me guess, does the comment appear at random places in the generated output? :-)

I noticed very strange distributions of comments before. I could imagine that this is caused by the syn crate, which we use to parse the input into an ast. But didn't investigate it, because it didn't cause any trouble.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It seems to consistently get moved into the build method just before the uninitialized field error string for "name", even when I move it around in the raw source.

// the mutable builder pattern.
LoremBuilder::default()
.name(name)
.try_addr(addr).map_err(|e| e.to_string())?
.build()
}
28 changes: 28 additions & 0 deletions derive_builder/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -216,6 +216,31 @@
//! }
//! ```
//!
//! ## Fallible Setters
//!
//! Alongside the normal setter methods, you can expose fallible setters which are generic over
//! the `TryInto` trait. TryInto is a not-yet-stable trait
//! (see rust-lang issue [#33417](https://github.com/rust-lang/rust/issues/33417)) similar to
//! `Into` with the key distinction that the conversion can fail, and therefore produces a `Result`.
//!
//! You can declare the `try_setter` attribute today, and your builder will work on stable Rust -
//! the fallible setters will automatically light up in future when the `try_from` feature is
//! stabilized.
//!
//!
//! ```rust,ignore
//! # #[macro_use]
//! # extern crate derive_builder;
//! #
//! #[derive(Builder, Debug, PartialEq)]
//! #[builder(try_setter, setter(into))]
//! struct Lorem {
//! pub ipsum: u8,
//! }
//! ```
//!
//! A complete example of using fallible setters is available in `examples/try_setter.rs`.
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

please add a note that the name and visibility are inherited from the setter settings.

You could change the example into something like this, to better explain it?

//! # #[macro_use]
//! # extern crate derive_builder;
//! #
//! #[derive(Builder, Debug, PartialEq)]
//! struct Lorem {
//!     #[builder(try_setter, setter(name="foo"))]
//!     pub ipsum: u8,
//! }

//!
//! ## Default Values
//!
//! You can define default values for each field via annotation by `#[builder(default="...")]`,
Expand Down Expand Up @@ -366,6 +391,9 @@
//! - If derive_builder depends on your crate, and vice versa, then a cyclic
//! dependency would occur. To break it you could try to depend on the
//! [`derive_builder_core`] crate instead.
//! - The `try_setter` attribute and `owned` builder pattern are not compatible in practice;
//! an error during building will consume the builder, making it impossible to continue
//! construction.
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nice catch :-)

//!
//! ## Debugging Info
//!
Expand Down
66 changes: 60 additions & 6 deletions derive_builder_core/src/setter.rs
Original file line number Diff line number Diff line change
Expand Up @@ -131,9 +131,9 @@ impl<'a> ToTokens for Setter<'a> {
#vis fn #try_ident #try_ty_params (#self_param, value: VALUE)
-> #result<#return_ty, VALUE::Err>
{
let value : #ty = value.try_into()?;
let converted : #ty = value.try_into()?;
let mut new = #self_into_return_ty;
new.#field_ident = #option::Some(value);
new.#field_ident = #option::Some(converted);
Ok(new)
}));
} else {
Expand Down Expand Up @@ -244,7 +244,7 @@ mod tests {
));
}

// including
// including try_setter
#[test]
fn full() {
let attrs = vec![syn::parse_outer_attr("#[some_attr]").unwrap()];
Expand All @@ -256,14 +256,36 @@ mod tests {
setter.attrs = attrs.as_slice();
setter.generic_into = true;
setter.deprecation_notes = &deprecated;
setter.try_setter = true;

// Had to hoist these out to avoid a recursion limit in the quote!
// macro invocation below.
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

no, problem. You can just add #![cfg_attr(test, recursion_limit="100")] to the crate root. That should increase the recursion limit in test-mode only. Just put in a number that's roughly twice as high as the current minimum requirement. :-)

let action = quote!(
let mut new = self;
new.foo = ::std::option::Option::Some(value.into());
new
);

let try_action = quote!(
let converted : Foo = value.try_into()?;
let mut new = self;
new.foo = ::std::option::Option::Some(converted);
Ok(new)
);

println!("{}", quote!(#setter));

assert_eq!(quote!(#setter), quote!(
#[some_attr]
pub fn foo <VALUE: ::std::convert::Into<Foo>>(&mut self, value: VALUE) -> &mut Self {
#deprecated
let mut new = self;
new.foo = ::std::option::Option::Some(value.into());
new
#action
}

#[some_attr]
pub fn try_foo<VALUE: ::std::convert::TryInto<Foo>>(&mut self, value: VALUE)
-> ::std::result::Result<&mut Self, VALUE::Err> {
#try_action
}
));
}
Expand Down Expand Up @@ -305,4 +327,36 @@ mod tests {

assert_eq!(quote!(#setter), quote!());
}

#[test]
fn try_setter() {
let mut setter : Setter = default_setter!();
setter.pattern = BuilderPattern::Mutable;
setter.try_setter = true;

let setter_quote = quote!(
pub fn foo(&mut self, value: Foo) -> &mut Self {
let mut new = self;
new.foo = ::std::option::Option::Some(value);
new
}
);

// hoisting necessary to avoid recursion limit.
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

same here

let try_setter_body = quote!(
let converted : Foo = value.try_into()?;
let mut new = self;
new.foo = ::std::option::Option::Some(converted);
Ok(new)
);

assert_eq!(quote!(#setter), quote!(
#setter_quote

pub fn try_foo<VALUE: ::std::convert::TryInto<Foo>>(&mut self, value: VALUE)
-> ::std::result::Result<&mut Self, VALUE::Err> {
#try_setter_body
}
));
}
}
2 changes: 1 addition & 1 deletion derive_builder_test/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ publish = false
path = "src/lib.rs"

[features]
compiletests = ["compiletest_rs"]
nightlytests = ["compiletest_rs"]
logging = [ "derive_builder/logging" ]
struct_default = [ "derive_builder/struct_default" ]

Expand Down
2 changes: 1 addition & 1 deletion derive_builder_test/tests/compiletests.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
#![cfg(feature = "compiletests")]
#![cfg(feature = "nightlytests")]
extern crate compiletest_rs as compiletest;

// note:
Expand Down
Original file line number Diff line number Diff line change
@@ -1,11 +1,11 @@
#![cfg_attr(feature = "try_setter", feature(try_from))]
#![cfg_attr(feature = "nightlytests", feature(try_from))]

#[macro_use]
extern crate derive_builder;

#[allow(unused_imports)]
mod struct_level {
#[cfg(feature = "try_setter")]
#[cfg(feature = "try_from")]
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ok, gating on try_from instead of nightlytests is actually a good idea here. Please do it consistently in the rest of the test then - that'd be easier to understand. :-)

Alternatively also gate on nightlytests here.

use std::convert::TryFrom;

use std::net::{IpAddr, AddrParseError};
Expand All @@ -21,7 +21,7 @@ mod struct_level {
}
}

#[cfg(feature = "try_setter")]
#[cfg(feature = "nightlytests")]
impl<'a> TryFrom<&'a str> for MyAddr {
type Err = AddrParseError;

Expand All @@ -39,11 +39,14 @@ mod struct_level {

#[test]
fn infallible_set() {
let _ = LoremBuilder::default().source(IpAddr::from_str("1.2.3.4").unwrap()).build();
let _ = LoremBuilder::default()
.source(IpAddr::from_str("1.2.3.4").unwrap())
.dest(IpAddr::from_str("0.0.0.0").unwrap())
.build();
}

#[test]
#[cfg(feature = "try_setter")]
#[cfg(feature = "nightlytests")]
fn fallible_set() {
let mut builder = LoremBuilder::default();
let try_result = builder.try_source("1.2.3.4");
Expand All @@ -55,15 +58,15 @@ mod struct_level {
}

// Allow dead code here since the test that uses this depends on the try_setter feature.
#[cfg_attr(not(feature = "try_setter"), allow(dead_code))]
#[cfg_attr(not(feature = "nightlytests"), allow(dead_code))]
fn exact_helper() -> Result<Lorem, String> {
LoremBuilder::default()
.source(IpAddr::from_str("1.2.3.4").unwrap())
.dest(IpAddr::from_str("0.0.0.0").unwrap())
.build()
}

#[cfg(feature = "try_setter")]
#[cfg(feature = "nightlytests")]
fn try_helper() -> Result<Lorem, String> {
LoremBuilder::default()
.try_source("1.2.3.4").map_err(|e| e.to_string())?
Expand All @@ -72,7 +75,7 @@ mod struct_level {
}

#[test]
#[cfg(feature = "try_setter")]
#[cfg(feature = "nightlytests")]
fn with_helper() {
assert_eq!(exact_helper().unwrap(), try_helper().unwrap());
}
Expand Down