mediaframe 0.1.3

A common media-stream descriptor vocabulary (pixel-format, colour, and frame metadata for video — audio/subtitle to follow) for media processing pipelines.
Documentation
//! EXIF capture-device metadata — make/model of the camera (or other
//! capture device) that produced a media file.
//!
//! Mirrors the long-standing `MediaMeta.device_make` / `device_model`
//! free-form `SmolStr` fields on findit-proto: a pair of small inline
//! strings holding e.g. `"Apple"` / `"iPhone 15 Pro"` or `"Sony"` /
//! `"ILCE-7M4"`. Empty string means absent (never `Option<SmolStr>`)
//! per the mediaframe convention shared with the codec module.

use smol_str::SmolStr;

/// EXIF-style capture device descriptor — manufacturer + model.
///
/// Sourced from EXIF tags `Make` (`0x010f`) / `Model` (`0x0110`) on
/// still images and from `com.apple.quicktime.make` /
/// `com.apple.quicktime.model` (and equivalent vendor) atoms on
/// MOV/MP4 video, as well as findit-proto's `MediaMeta.device_*`
/// `SmolStr` fields.
///
/// Both fields are private `SmolStr`s — the empty string is the
/// sentinel for "absent" so callers never need `Option<SmolStr>`
/// (matches the codec / source-tagging convention elsewhere in this
/// crate). Use [`Self::is_empty`] to detect the fully-absent state.
// `serde(default)` keeps sparse / older-schema JSON deserializable: missing
// fields fall back to the type-level `Default` impl (`Device::new()` — both
// `make` and `model` empty), matching the empty-string-means-absent convention.
#[cfg_attr(
  feature = "serde",
  derive(serde::Serialize, serde::Deserialize),
  serde(default)
)]
#[cfg_attr(
  feature = "quickcheck",
  derive(::quickcheck_richderive::Arbitrary),
  quickcheck(arbitrary = "crate::quickcheck_helpers::composite::capture_device")
)]
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct Device {
  make: SmolStr,
  model: SmolStr,
}

impl Default for Device {
  /// Delegates to [`Device::new`] — both `make` and `model` empty.
  #[cfg_attr(not(tarpaulin), inline(always))]
  fn default() -> Self {
    Self::new()
  }
}

impl Device {
  /// Constructs an all-empty `Device` (both `make` and `model` set
  /// to the empty string).
  ///
  /// The empty string is the sentinel for "absent"; callers use
  /// [`Self::with_make`] / [`Self::with_model`] (or the setters) to
  /// populate either side.
  #[cfg_attr(not(tarpaulin), inline(always))]
  pub const fn new() -> Self {
    Self {
      make: SmolStr::new_static(""),
      model: SmolStr::new_static(""),
    }
  }

  /// Returns the manufacturer (camera "make"), e.g. `"Apple"` /
  /// `"Sony"`. An empty string means absent.
  #[cfg_attr(not(tarpaulin), inline(always))]
  pub fn make(&self) -> &str {
    self.make.as_str()
  }

  /// Returns the camera model, e.g. `"iPhone 15 Pro"` / `"ILCE-7M4"`.
  /// An empty string means absent.
  #[cfg_attr(not(tarpaulin), inline(always))]
  pub fn model(&self) -> &str {
    self.model.as_str()
  }

  /// Sets the manufacturer (consuming builder).
  #[must_use]
  #[cfg_attr(not(tarpaulin), inline(always))]
  pub fn with_make(mut self, make: impl Into<SmolStr>) -> Self {
    self.make = make.into();
    self
  }

  /// Sets the manufacturer in place.
  #[cfg_attr(not(tarpaulin), inline(always))]
  pub fn set_make(&mut self, make: impl Into<SmolStr>) -> &mut Self {
    self.make = make.into();
    self
  }

  /// Sets the camera model (consuming builder).
  #[must_use]
  #[cfg_attr(not(tarpaulin), inline(always))]
  pub fn with_model(mut self, model: impl Into<SmolStr>) -> Self {
    self.model = model.into();
    self
  }

  /// Sets the camera model in place.
  #[cfg_attr(not(tarpaulin), inline(always))]
  pub fn set_model(&mut self, model: impl Into<SmolStr>) -> &mut Self {
    self.model = model.into();
    self
  }

  /// Returns `true` when both `make` and `model` are empty — i.e. no
  /// capture-device metadata is recorded.
  #[cfg_attr(not(tarpaulin), inline(always))]
  pub fn is_empty(&self) -> bool {
    self.make.is_empty() && self.model.is_empty()
  }
}

#[cfg(test)]
mod tests {
  use super::*;

  #[test]
  fn new_is_all_empty() {
    let d = Device::new();
    assert_eq!(d.make(), "");
    assert_eq!(d.model(), "");
    assert!(d.is_empty());
  }

  #[test]
  fn default_matches_new() {
    assert_eq!(Device::default(), Device::new());
  }

  #[test]
  fn builder_chain_populates() {
    let d = Device::new().with_make("Apple").with_model("iPhone 15 Pro");
    assert_eq!(d.make(), "Apple");
    assert_eq!(d.model(), "iPhone 15 Pro");
    assert!(!d.is_empty());
  }

  #[test]
  fn setters_mutate_in_place() {
    let mut d = Device::new();
    d.set_make("Sony");
    d.set_model("ILCE-7M4");
    assert_eq!(d.make(), "Sony");
    assert_eq!(d.model(), "ILCE-7M4");
    assert!(!d.is_empty());
  }

  #[test]
  fn is_empty_partial() {
    let m = Device::new().with_make("Apple");
    assert!(!m.is_empty());
    let n = Device::new().with_model("ILCE-7M4");
    assert!(!n.is_empty());
  }
}