use crate::{
client::Client,
error::Error as HttpError,
request::{validate, Pending, Request},
routing::Route,
};
use std::{
error::Error,
fmt::{Display, Formatter, Result as FmtResult},
};
use twilight_model::{
guild::audit_log::{AuditLog, AuditLogEvent},
id::{GuildId, UserId},
};
#[derive(Debug)]
pub struct GetAuditLogError {
kind: GetAuditLogErrorType,
}
impl GetAuditLogError {
#[must_use = "retrieving the type has no effect if left unused"]
pub const fn kind(&self) -> &GetAuditLogErrorType {
&self.kind
}
#[allow(clippy::unused_self)]
#[must_use = "consuming the error and retrieving the source has no effect if left unused"]
pub fn into_source(self) -> Option<Box<dyn Error + Send + Sync>> {
None
}
#[must_use = "consuming the error into its parts has no effect if left unused"]
pub fn into_parts(self) -> (GetAuditLogErrorType, Option<Box<dyn Error + Send + Sync>>) {
(self.kind, None)
}
}
impl Display for GetAuditLogError {
fn fmt(&self, f: &mut Formatter<'_>) -> FmtResult {
match &self.kind {
GetAuditLogErrorType::LimitInvalid { .. } => f.write_str("the limit is invalid"),
}
}
}
impl Error for GetAuditLogError {}
#[derive(Debug)]
#[non_exhaustive]
pub enum GetAuditLogErrorType {
LimitInvalid {
limit: u64,
},
}
#[derive(Default)]
struct GetAuditLogFields {
action_type: Option<AuditLogEvent>,
before: Option<u64>,
limit: Option<u64>,
user_id: Option<UserId>,
}
pub struct GetAuditLog<'a> {
fields: GetAuditLogFields,
fut: Option<Pending<'a, Option<AuditLog>>>,
guild_id: GuildId,
http: &'a Client,
}
impl<'a> GetAuditLog<'a> {
pub(crate) fn new(http: &'a Client, guild_id: GuildId) -> Self {
Self {
fields: GetAuditLogFields::default(),
fut: None,
guild_id,
http,
}
}
pub fn action_type(mut self, action_type: AuditLogEvent) -> Self {
self.fields.action_type.replace(action_type);
self
}
pub fn before(mut self, before: u64) -> Self {
self.fields.before.replace(before);
self
}
pub fn limit(mut self, limit: u64) -> Result<Self, GetAuditLogError> {
if !validate::get_audit_log_limit(limit) {
return Err(GetAuditLogError {
kind: GetAuditLogErrorType::LimitInvalid { limit },
});
}
self.fields.limit.replace(limit);
Ok(self)
}
pub fn user_id(mut self, user_id: UserId) -> Self {
self.fields.user_id.replace(user_id);
self
}
fn start(&mut self) -> Result<(), HttpError> {
let request = Request::from_route(Route::GetAuditLogs {
action_type: self.fields.action_type.map(|x| x as u64),
before: self.fields.before,
guild_id: self.guild_id.0,
limit: self.fields.limit,
user_id: self.fields.user_id.map(|x| x.0),
});
self.fut.replace(Box::pin(self.http.request(request)));
Ok(())
}
}
poll_req!(GetAuditLog<'_>, Option<AuditLog>);