utiles 0.8.0

Web map tile utils (aka utiles)
Documentation
use std::path::PathBuf;

use serde::Serialize;

use utiles_core::BBox;
use utiles_core::zoom::ZoomSet;

use crate::errors::UtilesCopyError;
use crate::errors::UtilesResult;
use crate::hash_types::HashType;
use crate::mbt::{MbtType, TilesFilter};
use crate::sqlite::InsertStrategy;

#[allow(clippy::struct_excessive_bools)]
#[derive(Debug, Clone, Serialize, Default)]
pub struct CopyConfig {
    pub src: PathBuf,
    pub dst: PathBuf,
    pub zset: Option<ZoomSet>,
    pub zooms: Option<Vec<u8>>,
    pub bboxes: Option<Vec<BBox>>,
    pub bounds_string: Option<String>,
    pub verbose: bool,
    pub dryrun: bool,
    pub force: bool,
    pub jobs: Option<u8>,
    pub istrat: InsertStrategy,
    pub dst_type: Option<MbtType>,
    pub hash: Option<HashType>,
    pub stream: bool,
}

impl CopyConfig {
    #[must_use]
    pub fn src_dbpath_str(&self) -> String {
        self.src.to_string_lossy().to_string()
    }

    pub fn mbtiles_sql_where(&self) -> UtilesResult<String> {
        let tf = TilesFilter::new(self.bboxes.clone(), self.zooms.clone());
        tf.mbtiles_sql_where(None)
    }

    pub fn tiles_stream_query(&self) -> UtilesResult<String> {
        let where_clause = self.mbtiles_sql_where()?;
        let sql = format!(
            "SELECT zoom_level, tile_column, tile_row, tile_data FROM tiles {where_clause}"
        );
        Ok(sql)
    }

    pub fn check_src_dst_same(&self) -> UtilesResult<()> {
        if self.src == self.dst {
            Err(
                UtilesCopyError::SrcDstSame(self.src.to_string_lossy().to_string())
                    .into(),
            )
        } else {
            Ok(())
        }
    }

    pub fn check_src_exists(&self) -> UtilesResult<()> {
        if self.src.exists() {
            Ok(())
        } else {
            Err(
                UtilesCopyError::SrcNotExists(format!("src: {}", self.src.display()))
                    .into(),
            )
        }
    }

    pub fn check(&self) -> UtilesResult<()> {
        self.check_src_exists()?;
        self.check_src_dst_same()?;
        Ok(())
    }

    #[must_use]
    pub fn njobs(&self) -> u8 {
        if let Some(j) = self.jobs {
            j
        } else {
            let ncpus = num_cpus::get();
            // if less than 4 cpus then use 1 job otherwise just default to 4 to
            // not throttle errything
            if ncpus < 4 { 1 } else { 4 }
        }
    }
}