|
1 |
| -use std::cell::RefCell; |
2 | 1 | use std::fmt;
|
| 2 | +use std::sync::Mutex; |
3 | 3 |
|
4 | 4 | use pyo3::exceptions::{PyTypeError, PyValueError};
|
5 | 5 | use pyo3::intern;
|
@@ -366,18 +366,27 @@ impl From<bool> for WarningsMode {
|
366 | 366 | }
|
367 | 367 | }
|
368 | 368 |
|
369 |
| -#[derive(Clone)] |
370 | 369 | #[cfg_attr(debug_assertions, derive(Debug))]
|
371 | 370 | pub(crate) struct CollectWarnings {
|
372 | 371 | mode: WarningsMode,
|
373 |
| - warnings: RefCell<Option<Vec<String>>>, |
| 372 | + // FIXME: mutex is to satisfy PyO3 0.23, we should be able to refactor this away |
| 373 | + warnings: Mutex<Vec<String>>, |
| 374 | +} |
| 375 | + |
| 376 | +impl Clone for CollectWarnings { |
| 377 | + fn clone(&self) -> Self { |
| 378 | + Self { |
| 379 | + mode: self.mode, |
| 380 | + warnings: Mutex::new(self.warnings.lock().expect("lock poisoned").clone()), |
| 381 | + } |
| 382 | + } |
374 | 383 | }
|
375 | 384 |
|
376 | 385 | impl CollectWarnings {
|
377 | 386 | pub(crate) fn new(mode: WarningsMode) -> Self {
|
378 | 387 | Self {
|
379 | 388 | mode,
|
380 |
| - warnings: RefCell::new(None), |
| 389 | + warnings: Mutex::new(Vec::new()), |
381 | 390 | }
|
382 | 391 | }
|
383 | 392 |
|
@@ -443,41 +452,46 @@ impl CollectWarnings {
|
443 | 452 | }
|
444 | 453 |
|
445 | 454 | fn add_warning(&self, message: String) {
|
446 |
| - let mut op_warnings = self.warnings.borrow_mut(); |
447 |
| - if let Some(ref mut warnings) = *op_warnings { |
448 |
| - warnings.push(message); |
449 |
| - } else { |
450 |
| - *op_warnings = Some(vec![message]); |
451 |
| - } |
| 455 | + self.warnings.lock().expect("lock poisoned").push(message) |
452 | 456 | }
|
453 | 457 |
|
454 | 458 | pub fn final_check(&self, py: Python) -> PyResult<()> {
|
455 | 459 | if self.mode == WarningsMode::None {
|
456 | 460 | return Ok(());
|
457 | 461 | }
|
458 |
| - match *self.warnings.borrow() { |
459 |
| - Some(ref warnings) => { |
460 |
| - let message = format!("Pydantic serializer warnings:\n {}", warnings.join("\n ")); |
461 |
| - if self.mode == WarningsMode::Warn { |
462 |
| - let user_warning_type = py.import_bound("builtins")?.getattr("UserWarning")?; |
463 |
| - PyErr::warn_bound(py, &user_warning_type, &message, 0) |
464 |
| - } else { |
465 |
| - Err(PydanticSerializationError::new_err(message)) |
466 |
| - } |
467 |
| - } |
468 |
| - _ => Ok(()), |
| 462 | + let warnings = self.warnings.lock().expect("lock poisoned"); |
| 463 | + |
| 464 | + if warnings.is_empty() { |
| 465 | + return Ok(()); |
| 466 | + } |
| 467 | + |
| 468 | + let message = format!("Pydantic serializer warnings:\n {}", warnings.join("\n ")); |
| 469 | + if self.mode == WarningsMode::Warn { |
| 470 | + let user_warning_type = py.import_bound("builtins")?.getattr("UserWarning")?; |
| 471 | + PyErr::warn_bound(py, &user_warning_type, &message, 0) |
| 472 | + } else { |
| 473 | + Err(PydanticSerializationError::new_err(message)) |
469 | 474 | }
|
470 | 475 | }
|
471 | 476 | }
|
472 | 477 |
|
473 |
| -#[derive(Default, Clone)] |
| 478 | +#[derive(Default)] |
474 | 479 | #[cfg_attr(debug_assertions, derive(Debug))]
|
475 | 480 | pub struct SerRecursionState {
|
476 |
| - guard: RefCell<RecursionState>, |
| 481 | + // FIXME: mutex is to satisfy PyO3 0.23, we should be able to refactor this away |
| 482 | + guard: Mutex<RecursionState>, |
| 483 | +} |
| 484 | + |
| 485 | +impl Clone for SerRecursionState { |
| 486 | + fn clone(&self) -> Self { |
| 487 | + Self { |
| 488 | + guard: Mutex::new(self.guard.lock().expect("lock poisoned").clone()), |
| 489 | + } |
| 490 | + } |
477 | 491 | }
|
478 | 492 |
|
479 | 493 | impl ContainsRecursionState for &'_ Extra<'_> {
|
480 | 494 | fn access_recursion_state<R>(&mut self, f: impl FnOnce(&mut RecursionState) -> R) -> R {
|
481 |
| - f(&mut self.rec_guard.guard.borrow_mut()) |
| 495 | + f(&mut self.rec_guard.guard.lock().expect("lock poisoned")) |
482 | 496 | }
|
483 | 497 | }
|
0 commit comments