diff --git a/crates/duckdb/CHANGELOG.md b/crates/duckdb/CHANGELOG.md index 18158f97..b596ff31 100644 --- a/crates/duckdb/CHANGELOG.md +++ b/crates/duckdb/CHANGELOG.md @@ -12,6 +12,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), - Read hive partitioned datasets, `Config` structure ([#624](https://github.com/stac-utils/stac-rs/pull/624)) - `Client.search_to_arrow_table` ([#634](https://github.com/stac-utils/stac-rs/pull/634)) - Conditionally disable parsing the WKB ([#635](https://github.com/stac-utils/stac-rs/pull/635)) +- `Client.extensions` ([#665](https://github.com/stac-utils/stac-rs/pull/665)) ### Removed diff --git a/crates/duckdb/src/lib.rs b/crates/duckdb/src/lib.rs index b48e81f0..25d6684c 100644 --- a/crates/duckdb/src/lib.rs +++ b/crates/duckdb/src/lib.rs @@ -112,6 +112,39 @@ pub struct Query { pub params: Vec, } +/// A DuckDB extension +// TODO implement aliases ... I don't know how vectors work yet 😢 +#[derive(Debug)] +pub struct Extension { + /// The extension name. + pub name: String, + + /// Is the extension loaded? + pub loaded: bool, + + /// Is the extension installed? + pub installed: bool, + + /// The path to the extension. + /// + /// This might be `(BUILT-IN)` for the core extensions. + pub install_path: Option, + + /// The extension description. + pub description: String, + + /// The extension version. + pub version: Option, + + /// The install mode. + /// + /// We don't bother making this an enum, yet. + pub install_mode: Option, + + /// Where the extension was installed from. + pub installed_from: Option, +} + impl Client { /// Creates a new client with no data sources. /// @@ -152,6 +185,39 @@ impl Client { Ok(Client { connection, config }) } + /// Returns a vector of all extensions. + /// + /// # Examples + /// + /// ``` + /// use stac_duckdb::Client; + /// + /// let client = Client::new().unwrap(); + /// let extensions = client.extensions().unwrap(); + /// let spatial = extensions.into_iter().find(|extension| extension.name == "spatial").unwrap(); + /// assert!(spatial.loaded); + /// ``` + pub fn extensions(&self) -> Result> { + let mut statement = self.connection.prepare( + "SELECT extension_name, loaded, installed, install_path, description, extension_version, install_mode, installed_from FROM duckdb_extensions();", + )?; + let extensions = statement + .query_map([], |row| { + Ok(Extension { + name: row.get("extension_name")?, + loaded: row.get("loaded")?, + installed: row.get("installed")?, + install_path: row.get("install_path")?, + description: row.get("description")?, + version: row.get("extension_version")?, + install_mode: row.get("install_mode")?, + installed_from: row.get("installed_from")?, + }) + })? + .collect::, duckdb::Error>>()?; + Ok(extensions) + } + /// Returns one or more [stac::Collection] from the items in the stac-geoparquet file. pub fn collections(&self, href: &str) -> Result> { let start_datetime= if self.connection.prepare(&format!(