#![doc(test(attr(deny(warnings))))]
#![doc(test(attr(allow(dead_code))))]
#![doc(test(attr(allow(unused_variables))))]
#![deny(broken_intra_doc_links)]
#![deny(invalid_codeblock_attributes)]
use std::convert::TryFrom;
use std::fmt;
use http::HeaderValue;
pub const MAX_PENDING_REQS: u32 = 16 * 1024;
pub const INVALID_REQUEST_HANDLE: u32 = std::u32::MAX - 1;
pub const INVALID_PENDING_REQUEST_HANDLE: u32 = std::u32::MAX - 1;
pub const INVALID_RESPONSE_HANDLE: u32 = std::u32::MAX - 1;
pub const INVALID_BODY_HANDLE: u32 = std::u32::MAX - 1;
pub const INVALID_DICTIONARY_HANDLE: u32 = std::u32::MAX - 1;
#[deprecated(since = "0.5.0", note = "renamed to FastlyStatus")]
pub type XqdStatus = FastlyStatus;
#[derive(Clone, Copy, Eq, PartialEq)]
#[repr(transparent)]
pub struct FastlyStatus {
pub code: i32,
}
impl FastlyStatus {
pub const OK: Self = Self { code: 0 };
pub const ERROR: Self = Self { code: 1 };
pub const INVAL: Self = Self { code: 2 };
pub const BADF: Self = Self { code: 3 };
pub const BUFLEN: Self = Self { code: 4 };
pub const UNSUPPORTED: Self = Self { code: 5 };
pub const BADALIGN: Self = Self { code: 6 };
#[deprecated(
since = "0.6.1",
note = "Please use the equivalent `HTTPINVALID` constant instead"
)]
pub const HTTPPARSE: Self = Self { code: 7 };
pub const HTTPINVALID: Self = Self { code: 7 };
pub const HTTPUSER: Self = Self { code: 8 };
pub const HTTPINCOMPLETE: Self = Self { code: 9 };
pub const NONE: Self = Self { code: 10 };
pub fn is_ok(&self) -> bool {
self == &Self::OK
}
pub fn is_err(&self) -> bool {
!self.is_ok()
}
pub fn result(self) -> Result<(), Self> {
if let Self::OK = self {
Ok(())
} else {
Err(self)
}
}
}
impl fmt::Debug for FastlyStatus {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.write_str(match *self {
Self::OK => "OK",
Self::ERROR => "ERROR",
Self::INVAL => "INVAL",
Self::BADF => "BADF",
Self::BUFLEN => "BUFLEN",
Self::BADALIGN => "BADALIGN",
Self::HTTPINVALID => "HTTP_INVALID_ERROR",
Self::HTTPUSER => "HTTP_USER_ERROR",
Self::HTTPINCOMPLETE => "HTTP_INCOMPLETE_MESSAGE",
Self::NONE => "NONE",
_ => "UNKNOWN",
})
}
}
#[deprecated(since = "0.5.0", note = "renamed to FASTLY_ABI_VERSION")]
pub const XQD_ABI_VERSION: u64 = FASTLY_ABI_VERSION;
pub const FASTLY_ABI_VERSION: u64 = 1;
#[derive(Clone, Copy, Eq, Hash, Ord, PartialEq, PartialOrd)]
#[repr(u32)]
pub enum HttpVersion {
Http09 = 0,
Http10 = 1,
Http11 = 2,
H2 = 3,
H3 = 4,
}
impl HttpVersion {
pub fn as_u32(&self) -> u32 {
*self as u32
}
}
impl TryFrom<u32> for HttpVersion {
type Error = String;
fn try_from(x: u32) -> Result<Self, Self::Error> {
if x == Self::Http09 as u32 {
Ok(Self::Http09)
} else if x == Self::Http10 as u32 {
Ok(Self::Http10)
} else if x == Self::Http11 as u32 {
Ok(Self::Http11)
} else if x == Self::H2 as u32 {
Ok(Self::H2)
} else if x == Self::H3 as u32 {
Ok(Self::H3)
} else {
Err(format!("unknown http version enum value: {}", x))
}
}
}
impl From<http::Version> for HttpVersion {
fn from(v: http::Version) -> Self {
match v {
http::Version::HTTP_09 => Self::Http09,
http::Version::HTTP_10 => Self::Http10,
http::Version::HTTP_11 => Self::Http11,
http::Version::HTTP_2 => Self::H2,
http::Version::HTTP_3 => Self::H3,
_ => unreachable!(),
}
}
}
impl From<HttpVersion> for http::Version {
fn from(v: HttpVersion) -> Self {
match v {
HttpVersion::Http09 => Self::HTTP_09,
HttpVersion::Http10 => Self::HTTP_10,
HttpVersion::Http11 => Self::HTTP_11,
HttpVersion::H2 => Self::HTTP_2,
HttpVersion::H3 => Self::HTTP_3,
}
}
}
#[derive(Clone, Copy, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
#[repr(u32)]
pub enum BodyWriteEnd {
Back = 0,
Front = 1,
}
#[derive(Clone, Debug)]
pub enum CacheOverride {
None,
Pass,
Override {
ttl: Option<u32>,
stale_while_revalidate: Option<u32>,
pci: bool,
surrogate_key: Option<HeaderValue>,
},
}
impl CacheOverride {
pub const fn none() -> Self {
Self::None
}
pub const fn pass() -> Self {
Self::Pass
}
pub fn is_pass(&self) -> bool {
if let Self::Pass = self {
true
} else {
false
}
}
pub const fn ttl(ttl: u32) -> Self {
Self::Override {
ttl: Some(ttl),
stale_while_revalidate: None,
pci: false,
surrogate_key: None,
}
}
pub const fn stale_while_revalidate(swr: u32) -> Self {
Self::Override {
ttl: None,
stale_while_revalidate: Some(swr),
pci: false,
surrogate_key: None,
}
}
pub const fn pci(pci: bool) -> Self {
Self::Override {
ttl: None,
stale_while_revalidate: None,
pci,
surrogate_key: None,
}
}
pub const fn surrogate_key(sk: HeaderValue) -> Self {
Self::Override {
ttl: None,
stale_while_revalidate: None,
pci: false,
surrogate_key: Some(sk),
}
}
pub fn set_none(&mut self) {
*self = Self::None;
}
pub fn set_pass(&mut self, pass: bool) {
if pass {
*self = Self::Pass;
} else if let Self::Pass = self {
*self = Self::None;
}
}
pub fn set_ttl(&mut self, new_ttl: u32) {
match self {
Self::Override { ttl, .. } => *ttl = Some(new_ttl),
_ => *self = Self::ttl(new_ttl),
}
}
pub fn set_stale_while_revalidate(&mut self, new_swr: u32) {
match self {
Self::Override {
stale_while_revalidate,
..
} => *stale_while_revalidate = Some(new_swr),
_ => *self = Self::stale_while_revalidate(new_swr),
}
}
pub fn set_pci(&mut self, new_pci: bool) {
match self {
Self::Override { pci, .. } => *pci = new_pci,
_ => *self = Self::pci(new_pci),
}
}
pub fn set_surrogate_key(&mut self, new_surrogate_key: HeaderValue) {
match self {
Self::Override { surrogate_key, .. } => *surrogate_key = Some(new_surrogate_key),
_ => *self = Self::surrogate_key(new_surrogate_key),
}
}
pub const fn default() -> Self {
Self::None
}
#[doc(hidden)]
pub fn to_abi(&self) -> (u32, u32, u32, Option<&[u8]>) {
match *self {
Self::None => (CacheOverrideTag::empty().bits(), 0, 0, None),
Self::Pass => (CacheOverrideTag::PASS.bits(), 0, 0, None),
Self::Override {
ttl,
stale_while_revalidate,
pci,
ref surrogate_key,
} => {
let mut tag = CacheOverrideTag::empty();
let ttl = if let Some(ttl) = ttl {
tag |= CacheOverrideTag::TTL;
ttl
} else {
0
};
let swr = if let Some(swr) = stale_while_revalidate {
tag |= CacheOverrideTag::STALE_WHILE_REVALIDATE;
swr
} else {
0
};
if pci {
tag |= CacheOverrideTag::PCI;
}
let sk = surrogate_key.as_ref().map(HeaderValue::as_bytes);
(tag.bits(), ttl, swr, sk)
}
}
}
#[doc(hidden)]
pub fn from_abi(
tag: u32,
ttl: u32,
swr: u32,
surrogate_key: Option<HeaderValue>,
) -> Option<Self> {
CacheOverrideTag::from_bits(tag).map(|tag| {
if tag.contains(CacheOverrideTag::PASS) {
return CacheOverride::Pass;
}
if tag.is_empty() && surrogate_key.is_none() {
return CacheOverride::None;
}
let ttl = if tag.contains(CacheOverrideTag::TTL) {
Some(ttl)
} else {
None
};
let stale_while_revalidate = if tag.contains(CacheOverrideTag::STALE_WHILE_REVALIDATE) {
Some(swr)
} else {
None
};
let pci = tag.contains(CacheOverrideTag::PCI);
CacheOverride::Override {
ttl,
stale_while_revalidate,
pci,
surrogate_key,
}
})
}
}
bitflags::bitflags! {
struct CacheOverrideTag: u32 {
const PASS = 1 << 0;
const TTL = 1 << 1;
const STALE_WHILE_REVALIDATE = 1 << 2;
const PCI = 1 << 3;
}
}