1
- use std:: borrow:: Cow ;
2
-
3
1
use pyo3:: intern;
4
2
use pyo3:: prelude:: * ;
5
3
use pyo3:: types:: PyDict ;
6
4
7
- use crate :: build_tools:: { py_err, SchemaDict } ;
8
- use crate :: errors:: { ValError , ValResult } ;
5
+ use crate :: build_tools:: { py_err, schema_or_config_same , SchemaDict } ;
6
+ use crate :: errors:: { LocItem , ValError , ValResult } ;
9
7
use crate :: input:: Input ;
10
8
use crate :: questions:: Question ;
11
9
use crate :: recursion_guard:: RecursionGuard ;
@@ -33,10 +31,10 @@ impl DefaultType {
33
31
}
34
32
}
35
33
36
- pub fn default_value ( & self , py : Python ) -> PyResult < Option < Cow < PyObject > > > {
34
+ pub fn default_value ( & self , py : Python ) -> PyResult < Option < PyObject > > {
37
35
match self {
38
- Self :: Default ( ref default) => Ok ( Some ( Cow :: Borrowed ( default) ) ) ,
39
- Self :: DefaultFactory ( ref default_factory) => Ok ( Some ( Cow :: Owned ( default_factory. call0 ( py) ?) ) ) ,
36
+ Self :: Default ( ref default) => Ok ( Some ( default. clone_ref ( py ) ) ) ,
37
+ Self :: DefaultFactory ( ref default_factory) => Ok ( Some ( default_factory. call0 ( py) ?) ) ,
40
38
Self :: None => Ok ( None ) ,
41
39
}
42
40
}
@@ -54,6 +52,7 @@ pub struct WithDefaultValidator {
54
52
default : DefaultType ,
55
53
on_error : OnError ,
56
54
validator : Box < CombinedValidator > ,
55
+ validate_default : bool ,
57
56
name : String ,
58
57
}
59
58
@@ -89,6 +88,7 @@ impl BuildValidator for WithDefaultValidator {
89
88
default,
90
89
on_error,
91
90
validator,
91
+ validate_default : schema_or_config_same ( schema, config, intern ! ( py, "validate_default" ) ) ?. unwrap_or ( false ) ,
92
92
name,
93
93
}
94
94
. into ( ) )
@@ -108,12 +108,43 @@ impl Validator for WithDefaultValidator {
108
108
Ok ( v) => Ok ( v) ,
109
109
Err ( e) => match self . on_error {
110
110
OnError :: Raise => Err ( e) ,
111
- OnError :: Default => Ok ( self . default . default_value ( py) ?. unwrap ( ) . as_ref ( ) . clone ( ) ) ,
111
+ OnError :: Default => Ok ( self
112
+ . default_value ( py, None :: < usize > , extra, slots, recursion_guard) ?
113
+ . unwrap ( ) ) ,
112
114
OnError :: Omit => Err ( ValError :: Omit ) ,
113
115
} ,
114
116
}
115
117
}
116
118
119
+ fn default_value < ' s , ' data > (
120
+ & ' s self ,
121
+ py : Python < ' data > ,
122
+ outer_loc : Option < impl Into < LocItem > > ,
123
+ extra : & Extra ,
124
+ slots : & ' data [ CombinedValidator ] ,
125
+ recursion_guard : & ' s mut RecursionGuard ,
126
+ ) -> ValResult < ' data , Option < PyObject > > {
127
+ match self . default . default_value ( py) ? {
128
+ Some ( dft) => {
129
+ if self . validate_default {
130
+ match self . validate ( py, dft. into_ref ( py) , extra, slots, recursion_guard) {
131
+ Ok ( v) => Ok ( Some ( v) ) ,
132
+ Err ( e) => {
133
+ if let Some ( outer_loc) = outer_loc {
134
+ Err ( e. with_outer_location ( outer_loc. into ( ) ) )
135
+ } else {
136
+ Err ( e)
137
+ }
138
+ }
139
+ }
140
+ } else {
141
+ Ok ( Some ( dft) )
142
+ }
143
+ }
144
+ None => Ok ( None ) ,
145
+ }
146
+ }
147
+
117
148
fn get_name ( & self ) -> & str {
118
149
& self . name
119
150
}
@@ -136,11 +167,3 @@ impl WithDefaultValidator {
136
167
matches ! ( self . on_error, OnError :: Omit )
137
168
}
138
169
}
139
-
140
- pub fn get_default < ' a > ( py : Python < ' a > , validator : & ' a CombinedValidator ) -> PyResult < Option < Cow < ' a , PyObject > > > {
141
- if let CombinedValidator :: WithDefault ( validator) = validator {
142
- validator. default . default_value ( py)
143
- } else {
144
- Ok ( None )
145
- }
146
- }
0 commit comments