1#![allow(missing_docs)]
8
9use std::fmt;
10use thiserror::Error;
11
12pub type Result<T> = std::result::Result<T, CodecError>;
14
15#[derive(Error, Debug)]
17pub enum CodecError {
18 #[error("Invalid codec configuration: {details}")]
20 InvalidConfig { details: String },
21
22 #[error("Unsupported codec type: {codec_type}")]
24 UnsupportedCodec { codec_type: String },
25
26 #[error("Invalid audio format: {details}")]
28 InvalidFormat { details: String },
29
30 #[error("Invalid frame size: expected {expected}, got {actual}")]
32 InvalidFrameSize { expected: usize, actual: usize },
33
34 #[error("Invalid sample rate: {rate}Hz (supported: {supported:?})")]
36 InvalidSampleRate { rate: u32, supported: Vec<u32> },
37
38 #[error("Invalid channel count: {channels} (supported: {supported:?})")]
40 InvalidChannelCount { channels: u8, supported: Vec<u8> },
41
42 #[error("Invalid bitrate: {bitrate}bps (range: {min}-{max})")]
44 InvalidBitrate { bitrate: u32, min: u32, max: u32 },
45
46 #[error("Encoding failed: {reason}")]
48 EncodingFailed { reason: String },
49
50 #[error("Decoding failed: {reason}")]
52 DecodingFailed { reason: String },
53
54 #[error("Buffer too small: need {needed} bytes, got {actual}")]
56 BufferTooSmall { needed: usize, actual: usize },
57
58 #[error("Buffer overflow: attempted to write {size} bytes to {capacity} byte buffer")]
60 BufferOverflow { size: usize, capacity: usize },
61
62 #[error("Codec initialization failed: {reason}")]
64 InitializationFailed { reason: String },
65
66 #[error("Codec reset failed: {reason}")]
68 ResetFailed { reason: String },
69
70 #[error("Invalid payload data: {details}")]
72 InvalidPayload { details: String },
73
74 #[error("Codec not found: {name}")]
76 CodecNotFound { name: String },
77
78 #[error("Feature not enabled: {feature} (enable with --features {feature})")]
80 FeatureNotEnabled { feature: String },
81
82 #[error("SIMD operation failed: {reason}")]
84 SimdFailed { reason: String },
85
86 #[error("Math operation failed: {operation} - {reason}")]
88 MathError { operation: String, reason: String },
89
90 #[error("I/O operation failed: {reason}")]
92 IoError { reason: String },
93
94 #[error("External library error: {library} - {error}")]
96 ExternalLibraryError { library: String, error: String },
97
98 #[error("Internal error: {message} (this is a bug, please report it)")]
100 InternalError { message: String },
101}
102
103impl CodecError {
104 pub fn invalid_config(details: impl Into<String>) -> Self {
106 Self::InvalidConfig {
107 details: details.into(),
108 }
109 }
110
111 pub fn unsupported_codec(codec_type: impl Into<String>) -> Self {
113 Self::UnsupportedCodec {
114 codec_type: codec_type.into(),
115 }
116 }
117
118 pub fn invalid_format(details: impl Into<String>) -> Self {
120 Self::InvalidFormat {
121 details: details.into(),
122 }
123 }
124
125 pub fn encoding_failed(reason: impl Into<String>) -> Self {
127 Self::EncodingFailed {
128 reason: reason.into(),
129 }
130 }
131
132 pub fn decoding_failed(reason: impl Into<String>) -> Self {
134 Self::DecodingFailed {
135 reason: reason.into(),
136 }
137 }
138
139 pub fn initialization_failed(reason: impl Into<String>) -> Self {
141 Self::InitializationFailed {
142 reason: reason.into(),
143 }
144 }
145
146 pub fn feature_not_enabled(feature: impl Into<String>) -> Self {
148 Self::FeatureNotEnabled {
149 feature: feature.into(),
150 }
151 }
152
153 pub fn internal_error(message: impl Into<String>) -> Self {
155 Self::InternalError {
156 message: message.into(),
157 }
158 }
159
160 pub fn is_recoverable(&self) -> bool {
162 match self {
163 Self::InvalidConfig { .. }
165 | Self::UnsupportedCodec { .. }
166 | Self::InvalidFormat { .. }
167 | Self::InvalidSampleRate { .. }
168 | Self::InvalidChannelCount { .. }
169 | Self::InvalidBitrate { .. }
170 | Self::FeatureNotEnabled { .. }
171 | Self::CodecNotFound { .. }
172 | Self::InternalError { .. } => false,
173
174 Self::InvalidFrameSize { .. }
176 | Self::EncodingFailed { .. }
177 | Self::DecodingFailed { .. }
178 | Self::BufferTooSmall { .. }
179 | Self::BufferOverflow { .. }
180 | Self::InvalidPayload { .. }
181 | Self::SimdFailed { .. }
182 | Self::MathError { .. }
183 | Self::IoError { .. }
184 | Self::ExternalLibraryError { .. } => true,
185
186 Self::InitializationFailed { .. } | Self::ResetFailed { .. } => false,
188 }
189 }
190
191 pub fn category(&self) -> ErrorCategory {
193 match self {
194 Self::InvalidConfig { .. }
195 | Self::UnsupportedCodec { .. }
196 | Self::InvalidFormat { .. }
197 | Self::InvalidSampleRate { .. }
198 | Self::InvalidChannelCount { .. }
199 | Self::InvalidBitrate { .. }
200 | Self::FeatureNotEnabled { .. }
201 | Self::CodecNotFound { .. } => ErrorCategory::Configuration,
202
203 Self::EncodingFailed { .. }
204 | Self::DecodingFailed { .. }
205 | Self::InvalidFrameSize { .. }
206 | Self::InvalidPayload { .. } => ErrorCategory::Processing,
207
208 Self::BufferTooSmall { .. } | Self::BufferOverflow { .. } => ErrorCategory::Memory,
209
210 Self::InitializationFailed { .. } | Self::ResetFailed { .. } => {
211 ErrorCategory::Initialization
212 }
213
214 Self::SimdFailed { .. } | Self::MathError { .. } => ErrorCategory::Computation,
215
216 Self::IoError { .. } => ErrorCategory::Io,
217
218 Self::ExternalLibraryError { .. } => ErrorCategory::External,
219
220 Self::InternalError { .. } => ErrorCategory::Internal,
221 }
222 }
223}
224
225#[derive(Debug, Clone, Copy, PartialEq, Eq)]
227pub enum ErrorCategory {
228 Configuration,
230 Processing,
232 Memory,
234 Initialization,
236 Computation,
238 Io,
240 External,
242 Internal,
244}
245
246impl fmt::Display for ErrorCategory {
247 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
248 match self {
249 Self::Configuration => write!(f, "Configuration"),
250 Self::Processing => write!(f, "Processing"),
251 Self::Memory => write!(f, "Memory"),
252 Self::Initialization => write!(f, "Initialization"),
253 Self::Computation => write!(f, "Computation"),
254 Self::Io => write!(f, "I/O"),
255 Self::External => write!(f, "External"),
256 Self::Internal => write!(f, "Internal"),
257 }
258 }
259}
260
261impl From<std::io::Error> for CodecError {
263 fn from(error: std::io::Error) -> Self {
264 Self::IoError {
265 reason: error.to_string(),
266 }
267 }
268}
269
270impl From<std::num::ParseIntError> for CodecError {
272 fn from(error: std::num::ParseIntError) -> Self {
273 Self::MathError {
274 operation: "parse_int".to_string(),
275 reason: error.to_string(),
276 }
277 }
278}
279
280impl From<std::num::ParseFloatError> for CodecError {
282 fn from(error: std::num::ParseFloatError) -> Self {
283 Self::MathError {
284 operation: "parse_float".to_string(),
285 reason: error.to_string(),
286 }
287 }
288}
289
290impl From<&str> for CodecError {
291 fn from(s: &str) -> Self {
292 Self::InvalidConfig {
293 details: s.to_string(),
294 }
295 }
296}
297
298impl From<String> for CodecError {
299 fn from(s: String) -> Self {
300 Self::InvalidConfig { details: s }
301 }
302}
303
304#[cfg(test)]
305mod tests {
306 use super::*;
307
308 #[test]
309 fn test_error_creation() {
310 let err = CodecError::invalid_config("test message");
311 assert!(matches!(err, CodecError::InvalidConfig { .. }));
312 assert_eq!(err.category(), ErrorCategory::Configuration);
313 }
314
315 #[test]
316 fn test_error_recoverability() {
317 let recoverable = CodecError::EncodingFailed {
318 reason: "test".to_string(),
319 };
320 assert!(recoverable.is_recoverable());
321
322 let non_recoverable = CodecError::InvalidConfig {
323 details: "test".to_string(),
324 };
325 assert!(!non_recoverable.is_recoverable());
326 }
327
328 #[test]
329 fn test_error_categories() {
330 assert_eq!(
331 CodecError::invalid_config("test").category(),
332 ErrorCategory::Configuration
333 );
334 assert_eq!(
335 CodecError::encoding_failed("test").category(),
336 ErrorCategory::Processing
337 );
338 assert_eq!(
339 CodecError::BufferTooSmall {
340 needed: 100,
341 actual: 50
342 }
343 .category(),
344 ErrorCategory::Memory
345 );
346 }
347
348 #[test]
349 fn test_error_display() {
350 let err = CodecError::InvalidFrameSize {
351 expected: 160,
352 actual: 80,
353 };
354 let display = format!("{}", err);
355 assert!(display.contains("expected 160"));
356 assert!(display.contains("got 80"));
357 }
358
359 #[test]
360 fn test_error_conversion() {
361 let io_err = std::io::Error::new(std::io::ErrorKind::NotFound, "file not found");
362 let codec_err: CodecError = io_err.into();
363 assert!(matches!(codec_err, CodecError::IoError { .. }));
364 }
365}