stripe_shared/
dispute.rs

1/// A dispute occurs when a customer questions your charge with their card issuer.
2/// When this happens, you have the opportunity to respond to the dispute with
3/// evidence that shows that the charge is legitimate.
4///
5/// Related guide: [Disputes and fraud](https://stripe.com/docs/disputes)
6///
7/// For more details see <<https://stripe.com/docs/api/disputes/object>>.
8#[derive(Clone, Debug)]
9#[cfg_attr(feature = "deserialize", derive(serde::Deserialize))]
10pub struct Dispute {
11    /// Disputed amount.
12    /// Usually the amount of the charge, but it can differ (usually because of currency fluctuation or because only part of the order is disputed).
13    pub amount: i64,
14    /// List of zero, one, or two balance transactions that show funds withdrawn and reinstated to your Stripe account as a result of this dispute.
15    pub balance_transactions: Vec<stripe_shared::BalanceTransaction>,
16    /// ID of the charge that's disputed.
17    pub charge: stripe_types::Expandable<stripe_shared::Charge>,
18    /// Time at which the object was created. Measured in seconds since the Unix epoch.
19    pub created: stripe_types::Timestamp,
20    /// Three-letter [ISO currency code](https://www.iso.org/iso-4217-currency-codes.html), in lowercase.
21    /// Must be a [supported currency](https://stripe.com/docs/currencies).
22    pub currency: stripe_types::Currency,
23    /// List of eligibility types that are included in `enhanced_evidence`.
24    pub enhanced_eligibility_types: Vec<DisputeEnhancedEligibilityTypes>,
25    pub evidence: stripe_shared::DisputeEvidence,
26    pub evidence_details: stripe_shared::DisputeEvidenceDetails,
27    /// Unique identifier for the object.
28    pub id: stripe_shared::DisputeId,
29    /// If true, it's still possible to refund the disputed payment.
30    /// After the payment has been fully refunded, no further funds are withdrawn from your Stripe account as a result of this dispute.
31    pub is_charge_refundable: bool,
32    /// Has the value `true` if the object exists in live mode or the value `false` if the object exists in test mode.
33    pub livemode: bool,
34    /// Set of [key-value pairs](https://stripe.com/docs/api/metadata) that you can attach to an object.
35    /// This can be useful for storing additional information about the object in a structured format.
36    pub metadata: std::collections::HashMap<String, String>,
37    /// Network-dependent reason code for the dispute.
38    pub network_reason_code: Option<String>,
39    /// ID of the PaymentIntent that's disputed.
40    pub payment_intent: Option<stripe_types::Expandable<stripe_shared::PaymentIntent>>,
41    pub payment_method_details: Option<stripe_shared::DisputePaymentMethodDetails>,
42    /// Reason given by cardholder for dispute.
43    /// Possible values are `bank_cannot_process`, `check_returned`, `credit_not_processed`, `customer_initiated`, `debit_not_authorized`, `duplicate`, `fraudulent`, `general`, `incorrect_account_details`, `insufficient_funds`, `product_not_received`, `product_unacceptable`, `subscription_canceled`, or `unrecognized`.
44    /// Learn more about [dispute reasons](https://stripe.com/docs/disputes/categories).
45    pub reason: String,
46    /// Current status of dispute.
47    /// Possible values are `warning_needs_response`, `warning_under_review`, `warning_closed`, `needs_response`, `under_review`, `won`, or `lost`.
48    pub status: DisputeStatus,
49}
50#[doc(hidden)]
51pub struct DisputeBuilder {
52    amount: Option<i64>,
53    balance_transactions: Option<Vec<stripe_shared::BalanceTransaction>>,
54    charge: Option<stripe_types::Expandable<stripe_shared::Charge>>,
55    created: Option<stripe_types::Timestamp>,
56    currency: Option<stripe_types::Currency>,
57    enhanced_eligibility_types: Option<Vec<DisputeEnhancedEligibilityTypes>>,
58    evidence: Option<stripe_shared::DisputeEvidence>,
59    evidence_details: Option<stripe_shared::DisputeEvidenceDetails>,
60    id: Option<stripe_shared::DisputeId>,
61    is_charge_refundable: Option<bool>,
62    livemode: Option<bool>,
63    metadata: Option<std::collections::HashMap<String, String>>,
64    network_reason_code: Option<Option<String>>,
65    payment_intent: Option<Option<stripe_types::Expandable<stripe_shared::PaymentIntent>>>,
66    payment_method_details: Option<Option<stripe_shared::DisputePaymentMethodDetails>>,
67    reason: Option<String>,
68    status: Option<DisputeStatus>,
69}
70
71#[allow(
72    unused_variables,
73    irrefutable_let_patterns,
74    clippy::let_unit_value,
75    clippy::match_single_binding,
76    clippy::single_match
77)]
78const _: () = {
79    use miniserde::de::{Map, Visitor};
80    use miniserde::json::Value;
81    use miniserde::{make_place, Deserialize, Result};
82    use stripe_types::miniserde_helpers::FromValueOpt;
83    use stripe_types::{MapBuilder, ObjectDeser};
84
85    make_place!(Place);
86
87    impl Deserialize for Dispute {
88        fn begin(out: &mut Option<Self>) -> &mut dyn Visitor {
89            Place::new(out)
90        }
91    }
92
93    struct Builder<'a> {
94        out: &'a mut Option<Dispute>,
95        builder: DisputeBuilder,
96    }
97
98    impl Visitor for Place<Dispute> {
99        fn map(&mut self) -> Result<Box<dyn Map + '_>> {
100            Ok(Box::new(Builder { out: &mut self.out, builder: DisputeBuilder::deser_default() }))
101        }
102    }
103
104    impl MapBuilder for DisputeBuilder {
105        type Out = Dispute;
106        fn key(&mut self, k: &str) -> Result<&mut dyn Visitor> {
107            Ok(match k {
108                "amount" => Deserialize::begin(&mut self.amount),
109                "balance_transactions" => Deserialize::begin(&mut self.balance_transactions),
110                "charge" => Deserialize::begin(&mut self.charge),
111                "created" => Deserialize::begin(&mut self.created),
112                "currency" => Deserialize::begin(&mut self.currency),
113                "enhanced_eligibility_types" => {
114                    Deserialize::begin(&mut self.enhanced_eligibility_types)
115                }
116                "evidence" => Deserialize::begin(&mut self.evidence),
117                "evidence_details" => Deserialize::begin(&mut self.evidence_details),
118                "id" => Deserialize::begin(&mut self.id),
119                "is_charge_refundable" => Deserialize::begin(&mut self.is_charge_refundable),
120                "livemode" => Deserialize::begin(&mut self.livemode),
121                "metadata" => Deserialize::begin(&mut self.metadata),
122                "network_reason_code" => Deserialize::begin(&mut self.network_reason_code),
123                "payment_intent" => Deserialize::begin(&mut self.payment_intent),
124                "payment_method_details" => Deserialize::begin(&mut self.payment_method_details),
125                "reason" => Deserialize::begin(&mut self.reason),
126                "status" => Deserialize::begin(&mut self.status),
127
128                _ => <dyn Visitor>::ignore(),
129            })
130        }
131
132        fn deser_default() -> Self {
133            Self {
134                amount: Deserialize::default(),
135                balance_transactions: Deserialize::default(),
136                charge: Deserialize::default(),
137                created: Deserialize::default(),
138                currency: Deserialize::default(),
139                enhanced_eligibility_types: Deserialize::default(),
140                evidence: Deserialize::default(),
141                evidence_details: Deserialize::default(),
142                id: Deserialize::default(),
143                is_charge_refundable: Deserialize::default(),
144                livemode: Deserialize::default(),
145                metadata: Deserialize::default(),
146                network_reason_code: Deserialize::default(),
147                payment_intent: Deserialize::default(),
148                payment_method_details: Deserialize::default(),
149                reason: Deserialize::default(),
150                status: Deserialize::default(),
151            }
152        }
153
154        fn take_out(&mut self) -> Option<Self::Out> {
155            let (
156                Some(amount),
157                Some(balance_transactions),
158                Some(charge),
159                Some(created),
160                Some(currency),
161                Some(enhanced_eligibility_types),
162                Some(evidence),
163                Some(evidence_details),
164                Some(id),
165                Some(is_charge_refundable),
166                Some(livemode),
167                Some(metadata),
168                Some(network_reason_code),
169                Some(payment_intent),
170                Some(payment_method_details),
171                Some(reason),
172                Some(status),
173            ) = (
174                self.amount,
175                self.balance_transactions.take(),
176                self.charge.take(),
177                self.created,
178                self.currency,
179                self.enhanced_eligibility_types.take(),
180                self.evidence.take(),
181                self.evidence_details.take(),
182                self.id.take(),
183                self.is_charge_refundable,
184                self.livemode,
185                self.metadata.take(),
186                self.network_reason_code.take(),
187                self.payment_intent.take(),
188                self.payment_method_details.take(),
189                self.reason.take(),
190                self.status,
191            )
192            else {
193                return None;
194            };
195            Some(Self::Out {
196                amount,
197                balance_transactions,
198                charge,
199                created,
200                currency,
201                enhanced_eligibility_types,
202                evidence,
203                evidence_details,
204                id,
205                is_charge_refundable,
206                livemode,
207                metadata,
208                network_reason_code,
209                payment_intent,
210                payment_method_details,
211                reason,
212                status,
213            })
214        }
215    }
216
217    impl<'a> Map for Builder<'a> {
218        fn key(&mut self, k: &str) -> Result<&mut dyn Visitor> {
219            self.builder.key(k)
220        }
221
222        fn finish(&mut self) -> Result<()> {
223            *self.out = self.builder.take_out();
224            Ok(())
225        }
226    }
227
228    impl ObjectDeser for Dispute {
229        type Builder = DisputeBuilder;
230    }
231
232    impl FromValueOpt for Dispute {
233        fn from_value(v: Value) -> Option<Self> {
234            let Value::Object(obj) = v else {
235                return None;
236            };
237            let mut b = DisputeBuilder::deser_default();
238            for (k, v) in obj {
239                match k.as_str() {
240                    "amount" => b.amount = FromValueOpt::from_value(v),
241                    "balance_transactions" => b.balance_transactions = FromValueOpt::from_value(v),
242                    "charge" => b.charge = FromValueOpt::from_value(v),
243                    "created" => b.created = FromValueOpt::from_value(v),
244                    "currency" => b.currency = FromValueOpt::from_value(v),
245                    "enhanced_eligibility_types" => {
246                        b.enhanced_eligibility_types = FromValueOpt::from_value(v)
247                    }
248                    "evidence" => b.evidence = FromValueOpt::from_value(v),
249                    "evidence_details" => b.evidence_details = FromValueOpt::from_value(v),
250                    "id" => b.id = FromValueOpt::from_value(v),
251                    "is_charge_refundable" => b.is_charge_refundable = FromValueOpt::from_value(v),
252                    "livemode" => b.livemode = FromValueOpt::from_value(v),
253                    "metadata" => b.metadata = FromValueOpt::from_value(v),
254                    "network_reason_code" => b.network_reason_code = FromValueOpt::from_value(v),
255                    "payment_intent" => b.payment_intent = FromValueOpt::from_value(v),
256                    "payment_method_details" => {
257                        b.payment_method_details = FromValueOpt::from_value(v)
258                    }
259                    "reason" => b.reason = FromValueOpt::from_value(v),
260                    "status" => b.status = FromValueOpt::from_value(v),
261
262                    _ => {}
263                }
264            }
265            b.take_out()
266        }
267    }
268};
269#[cfg(feature = "serialize")]
270impl serde::Serialize for Dispute {
271    fn serialize<S: serde::Serializer>(&self, s: S) -> Result<S::Ok, S::Error> {
272        use serde::ser::SerializeStruct;
273        let mut s = s.serialize_struct("Dispute", 18)?;
274        s.serialize_field("amount", &self.amount)?;
275        s.serialize_field("balance_transactions", &self.balance_transactions)?;
276        s.serialize_field("charge", &self.charge)?;
277        s.serialize_field("created", &self.created)?;
278        s.serialize_field("currency", &self.currency)?;
279        s.serialize_field("enhanced_eligibility_types", &self.enhanced_eligibility_types)?;
280        s.serialize_field("evidence", &self.evidence)?;
281        s.serialize_field("evidence_details", &self.evidence_details)?;
282        s.serialize_field("id", &self.id)?;
283        s.serialize_field("is_charge_refundable", &self.is_charge_refundable)?;
284        s.serialize_field("livemode", &self.livemode)?;
285        s.serialize_field("metadata", &self.metadata)?;
286        s.serialize_field("network_reason_code", &self.network_reason_code)?;
287        s.serialize_field("payment_intent", &self.payment_intent)?;
288        s.serialize_field("payment_method_details", &self.payment_method_details)?;
289        s.serialize_field("reason", &self.reason)?;
290        s.serialize_field("status", &self.status)?;
291
292        s.serialize_field("object", "dispute")?;
293        s.end()
294    }
295}
296/// List of eligibility types that are included in `enhanced_evidence`.
297#[derive(Copy, Clone, Eq, PartialEq)]
298pub enum DisputeEnhancedEligibilityTypes {
299    VisaCompellingEvidence3,
300}
301impl DisputeEnhancedEligibilityTypes {
302    pub fn as_str(self) -> &'static str {
303        use DisputeEnhancedEligibilityTypes::*;
304        match self {
305            VisaCompellingEvidence3 => "visa_compelling_evidence_3",
306        }
307    }
308}
309
310impl std::str::FromStr for DisputeEnhancedEligibilityTypes {
311    type Err = stripe_types::StripeParseError;
312    fn from_str(s: &str) -> Result<Self, Self::Err> {
313        use DisputeEnhancedEligibilityTypes::*;
314        match s {
315            "visa_compelling_evidence_3" => Ok(VisaCompellingEvidence3),
316            _ => Err(stripe_types::StripeParseError),
317        }
318    }
319}
320impl std::fmt::Display for DisputeEnhancedEligibilityTypes {
321    fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
322        f.write_str(self.as_str())
323    }
324}
325
326impl std::fmt::Debug for DisputeEnhancedEligibilityTypes {
327    fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
328        f.write_str(self.as_str())
329    }
330}
331#[cfg(feature = "serialize")]
332impl serde::Serialize for DisputeEnhancedEligibilityTypes {
333    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
334    where
335        S: serde::Serializer,
336    {
337        serializer.serialize_str(self.as_str())
338    }
339}
340impl miniserde::Deserialize for DisputeEnhancedEligibilityTypes {
341    fn begin(out: &mut Option<Self>) -> &mut dyn miniserde::de::Visitor {
342        crate::Place::new(out)
343    }
344}
345
346impl miniserde::de::Visitor for crate::Place<DisputeEnhancedEligibilityTypes> {
347    fn string(&mut self, s: &str) -> miniserde::Result<()> {
348        use std::str::FromStr;
349        self.out =
350            Some(DisputeEnhancedEligibilityTypes::from_str(s).map_err(|_| miniserde::Error)?);
351        Ok(())
352    }
353}
354
355stripe_types::impl_from_val_with_from_str!(DisputeEnhancedEligibilityTypes);
356#[cfg(feature = "deserialize")]
357impl<'de> serde::Deserialize<'de> for DisputeEnhancedEligibilityTypes {
358    fn deserialize<D: serde::Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
359        use std::str::FromStr;
360        let s: std::borrow::Cow<'de, str> = serde::Deserialize::deserialize(deserializer)?;
361        Self::from_str(&s).map_err(|_| {
362            serde::de::Error::custom("Unknown value for DisputeEnhancedEligibilityTypes")
363        })
364    }
365}
366/// Current status of dispute.
367/// Possible values are `warning_needs_response`, `warning_under_review`, `warning_closed`, `needs_response`, `under_review`, `won`, or `lost`.
368#[derive(Copy, Clone, Eq, PartialEq)]
369pub enum DisputeStatus {
370    Lost,
371    NeedsResponse,
372    UnderReview,
373    WarningClosed,
374    WarningNeedsResponse,
375    WarningUnderReview,
376    Won,
377}
378impl DisputeStatus {
379    pub fn as_str(self) -> &'static str {
380        use DisputeStatus::*;
381        match self {
382            Lost => "lost",
383            NeedsResponse => "needs_response",
384            UnderReview => "under_review",
385            WarningClosed => "warning_closed",
386            WarningNeedsResponse => "warning_needs_response",
387            WarningUnderReview => "warning_under_review",
388            Won => "won",
389        }
390    }
391}
392
393impl std::str::FromStr for DisputeStatus {
394    type Err = stripe_types::StripeParseError;
395    fn from_str(s: &str) -> Result<Self, Self::Err> {
396        use DisputeStatus::*;
397        match s {
398            "lost" => Ok(Lost),
399            "needs_response" => Ok(NeedsResponse),
400            "under_review" => Ok(UnderReview),
401            "warning_closed" => Ok(WarningClosed),
402            "warning_needs_response" => Ok(WarningNeedsResponse),
403            "warning_under_review" => Ok(WarningUnderReview),
404            "won" => Ok(Won),
405            _ => Err(stripe_types::StripeParseError),
406        }
407    }
408}
409impl std::fmt::Display for DisputeStatus {
410    fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
411        f.write_str(self.as_str())
412    }
413}
414
415impl std::fmt::Debug for DisputeStatus {
416    fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
417        f.write_str(self.as_str())
418    }
419}
420#[cfg(feature = "serialize")]
421impl serde::Serialize for DisputeStatus {
422    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
423    where
424        S: serde::Serializer,
425    {
426        serializer.serialize_str(self.as_str())
427    }
428}
429impl miniserde::Deserialize for DisputeStatus {
430    fn begin(out: &mut Option<Self>) -> &mut dyn miniserde::de::Visitor {
431        crate::Place::new(out)
432    }
433}
434
435impl miniserde::de::Visitor for crate::Place<DisputeStatus> {
436    fn string(&mut self, s: &str) -> miniserde::Result<()> {
437        use std::str::FromStr;
438        self.out = Some(DisputeStatus::from_str(s).map_err(|_| miniserde::Error)?);
439        Ok(())
440    }
441}
442
443stripe_types::impl_from_val_with_from_str!(DisputeStatus);
444#[cfg(feature = "deserialize")]
445impl<'de> serde::Deserialize<'de> for DisputeStatus {
446    fn deserialize<D: serde::Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
447        use std::str::FromStr;
448        let s: std::borrow::Cow<'de, str> = serde::Deserialize::deserialize(deserializer)?;
449        Self::from_str(&s).map_err(|_| serde::de::Error::custom("Unknown value for DisputeStatus"))
450    }
451}
452impl stripe_types::Object for Dispute {
453    type Id = stripe_shared::DisputeId;
454    fn id(&self) -> &Self::Id {
455        &self.id
456    }
457
458    fn into_id(self) -> Self::Id {
459        self.id
460    }
461}
462stripe_types::def_id!(DisputeId);