net_traits/
http_status.rs

1/* This Source Code Form is subject to the terms of the Mozilla Public
2 * License, v. 2.0. If a copy of the MPL was not distributed with this
3 * file, You can obtain one at https://mozilla.org/MPL/2.0/. */
4
5#![deny(unsafe_code)]
6
7use std::ops::RangeBounds;
8
9use serde::{Deserialize, Serialize};
10
11use crate::{MallocSizeOf, StatusCode};
12
13/// A representation of a HTTP Status Code and Message that can be used for
14/// DOM Response objects and other cases.
15/// These objects are immutable once created.
16#[derive(Clone, Debug, Deserialize, MallocSizeOf, PartialEq, Serialize)]
17pub struct HttpStatus {
18    code: u16,
19    message: Vec<u8>,
20}
21
22impl HttpStatus {
23    /// Creates a new HttpStatus for a valid status code.
24    pub fn new(code: StatusCode, message: Vec<u8>) -> Self {
25        Self {
26            code: code.as_u16(),
27            message,
28        }
29    }
30
31    /// Creates a new HttpStatus from a raw status code, but will panic
32    /// if the code is not in the 100 to 599 valid range.
33    pub fn new_raw(code: u16, message: Vec<u8>) -> Self {
34        if !(100..=599).contains(&code) {
35            panic!(
36                "HttpStatus code must be in the range 100 to 599, inclusive, but is {}",
37                code
38            );
39        }
40
41        Self { code, message }
42    }
43
44    /// Creates an instance that represents a Response.error() instance.
45    pub fn new_error() -> Self {
46        Self {
47            code: 0,
48            message: vec![],
49        }
50    }
51
52    /// Returns the StatusCode for non-error cases, panics otherwise.
53    pub fn code(&self) -> StatusCode {
54        StatusCode::from_u16(self.code).expect("HttpStatus code is 0, can't return a StatusCode")
55    }
56
57    /// Returns the StatusCode if not an error instance, or None otherwise.
58    pub fn try_code(&self) -> Option<StatusCode> {
59        StatusCode::from_u16(self.code).ok()
60    }
61
62    /// Returns the u16 representation of the access code. This is usable both for
63    /// valid HTTP status codes and in the error case.
64    pub fn raw_code(&self) -> u16 {
65        self.code
66    }
67
68    /// Get access to a reference of the message part.
69    pub fn message(&self) -> &[u8] {
70        &self.message
71    }
72
73    /// Helper that relays is_success() from the underlying code.
74    pub fn is_success(&self) -> bool {
75        StatusCode::from_u16(self.code).is_ok_and(|s| s.is_success())
76    }
77
78    /// True when the object was created with `new_error`.
79    pub fn is_error(&self) -> bool {
80        self.code == 0
81    }
82
83    /// Returns true if this status is in the given range.
84    /// Always return false for error statuses.
85    pub fn in_range<T: RangeBounds<u16>>(&self, range: T) -> bool {
86        self.code != 0 && range.contains(&self.code)
87    }
88}
89
90impl Default for HttpStatus {
91    /// The default implementation creates a "200 OK" response.
92    fn default() -> Self {
93        Self {
94            code: 200,
95            message: b"OK".to_vec(),
96        }
97    }
98}
99
100impl PartialEq<StatusCode> for HttpStatus {
101    fn eq(&self, other: &StatusCode) -> bool {
102        self.code == other.as_u16()
103    }
104}
105
106impl From<StatusCode> for HttpStatus {
107    fn from(code: StatusCode) -> Self {
108        Self {
109            code: code.as_u16(),
110            message: code
111                .canonical_reason()
112                .unwrap_or_default()
113                .as_bytes()
114                .to_vec(),
115        }
116    }
117}