mirror of
https://github.com/neondatabase/neon.git
synced 2026-01-14 08:52:56 +00:00
All Errors that can occur during get_installed_extensions() come from
tokio-postgres functions, e.g. if the database is being shut down
("FATAL: terminating connection due to administrator command"). I'm
seeing a lot of such errors in the logs with the regression tests, with
very verbose stack traces. The compute_ctl stack trace is pretty useless
for errors originating from the Postgres connection, the error message
has all the information, so stop printing the stack trace.
I changed the result type of the functions to return the originating
tokio_postgres Error rather than anyhow::Error, so that if we introduce
other error sources to the functions where the stack trace might be
useful, we'll be forced to revisit this, probably by introducing a new
Error type that separates postgres errors from other errors. But this
will do for now.
122 lines
3.9 KiB
Rust
122 lines
3.9 KiB
Rust
use std::collections::HashMap;
|
|
|
|
use anyhow::Result;
|
|
use compute_api::responses::{InstalledExtension, InstalledExtensions};
|
|
use tokio_postgres::error::Error as PostgresError;
|
|
use tokio_postgres::{Client, Config, NoTls};
|
|
|
|
use crate::metrics::INSTALLED_EXTENSIONS;
|
|
|
|
/// We don't reuse get_existing_dbs() just for code clarity
|
|
/// and to make database listing query here more explicit.
|
|
///
|
|
/// Limit the number of databases to 500 to avoid excessive load.
|
|
async fn list_dbs(client: &mut Client) -> Result<Vec<String>, PostgresError> {
|
|
// `pg_database.datconnlimit = -2` means that the database is in the
|
|
// invalid state
|
|
let databases = client
|
|
.query(
|
|
"SELECT datname FROM pg_catalog.pg_database
|
|
WHERE datallowconn
|
|
AND datconnlimit <> - 2
|
|
LIMIT 500",
|
|
&[],
|
|
)
|
|
.await?
|
|
.iter()
|
|
.map(|row| {
|
|
let db: String = row.get("datname");
|
|
db
|
|
})
|
|
.collect();
|
|
|
|
Ok(databases)
|
|
}
|
|
|
|
/// Connect to every database (see list_dbs above) and get the list of installed extensions.
|
|
///
|
|
/// Same extension can be installed in multiple databases with different versions,
|
|
/// so we report a separate metric (number of databases where it is installed)
|
|
/// for each extension version.
|
|
pub async fn get_installed_extensions(
|
|
mut conf: Config,
|
|
) -> Result<InstalledExtensions, PostgresError> {
|
|
conf.application_name("compute_ctl:get_installed_extensions");
|
|
let databases: Vec<String> = {
|
|
let (mut client, connection) = conf.connect(NoTls).await?;
|
|
tokio::spawn(async move {
|
|
if let Err(e) = connection.await {
|
|
eprintln!("connection error: {e}");
|
|
}
|
|
});
|
|
|
|
list_dbs(&mut client).await?
|
|
};
|
|
|
|
let mut extensions_map: HashMap<(String, String, String), InstalledExtension> = HashMap::new();
|
|
for db in databases.iter() {
|
|
conf.dbname(db);
|
|
|
|
let (client, connection) = conf.connect(NoTls).await?;
|
|
tokio::spawn(async move {
|
|
if let Err(e) = connection.await {
|
|
eprintln!("connection error: {e}");
|
|
}
|
|
});
|
|
|
|
let extensions: Vec<(String, String, i32)> = client
|
|
.query(
|
|
"SELECT extname, extversion, extowner::integer FROM pg_catalog.pg_extension",
|
|
&[],
|
|
)
|
|
.await?
|
|
.iter()
|
|
.map(|row| {
|
|
(
|
|
row.get("extname"),
|
|
row.get("extversion"),
|
|
row.get("extowner"),
|
|
)
|
|
})
|
|
.collect();
|
|
|
|
for (extname, v, extowner) in extensions.iter() {
|
|
let version = v.to_string();
|
|
|
|
// check if the extension is owned by superuser
|
|
// 10 is the oid of superuser
|
|
let owned_by_superuser = if *extowner == 10 { "1" } else { "0" };
|
|
|
|
extensions_map
|
|
.entry((
|
|
extname.to_string(),
|
|
version.clone(),
|
|
owned_by_superuser.to_string(),
|
|
))
|
|
.and_modify(|e| {
|
|
// count the number of databases where the extension is installed
|
|
e.n_databases += 1;
|
|
})
|
|
.or_insert(InstalledExtension {
|
|
extname: extname.to_string(),
|
|
version: version.clone(),
|
|
n_databases: 1,
|
|
owned_by_superuser: owned_by_superuser.to_string(),
|
|
});
|
|
}
|
|
}
|
|
|
|
for (key, ext) in extensions_map.iter() {
|
|
let (extname, version, owned_by_superuser) = key;
|
|
let n_databases = ext.n_databases as u64;
|
|
|
|
INSTALLED_EXTENSIONS
|
|
.with_label_values(&[extname, version, owned_by_superuser])
|
|
.set(n_databases);
|
|
}
|
|
|
|
Ok(InstalledExtensions {
|
|
extensions: extensions_map.into_values().collect(),
|
|
})
|
|
}
|