No meu caso, estou usando o AWS Redshift (baseado no Postgres). E parece que não há outras conexões com o banco de dados, mas estou recebendo esse mesmo erro.
ERROR: database "XYZ" is being accessed by other users
No meu caso, parece que o cluster do banco de dados ainda está processando no banco de dados e, embora não haja outras conexões externas / de usuários, o banco de dados ainda está em uso internamente. Encontrei isso executando o seguinte:
SELECT * FROM stv_sessions;
Então, meu truque foi escrever um loop no meu código, procurando linhas com o nome do meu banco de dados. (é claro que o loop não é infinito e é um loop sonolento, etc.)
SELECT * FROM stv_sessions where db_name = 'XYZ';
Se houver linhas, prossiga para excluir cada PID, um por um.
SELECT pg_terminate_backend(PUT_PID_HERE);
Se nenhuma linha for encontrada, prossiga para descartar o banco de dados
DROP DATABASE XYZ;
Nota: No meu caso, estou escrevendo testes de unidade / sistema Java, onde isso pode ser considerado aceitável. Isso não é aceitável para o código de produção.
Aqui está o hack completo, em Java (ignore minhas classes de teste / utilitário).
int i = 0;
while (i < 10) {
try {
i++;
logStandardOut("First try to delete session PIDs, before dropping the DB");
String getSessionPIDs = String.format("SELECT stv_sessions.process, stv_sessions.* FROM stv_sessions where db_name = '%s'", dbNameToReset);
ResultSet resultSet = databaseConnection.execQuery(getSessionPIDs);
while (resultSet.next()) {
int sessionPID = resultSet.getInt(1);
logStandardOut("killPID: %s", sessionPID);
String killSessionPID = String.format("select pg_terminate_backend(%s)", sessionPID);
try {
databaseConnection.execQuery(killSessionPID);
} catch (DatabaseException dbEx) {
//This is most commonly when a session PID is transient, where it ended between my query and kill lines
logStandardOut("Ignore it, you did your best: %s, %s", dbEx.getMessage(), dbEx.getCause());
}
}
//Drop the DB now
String dropDbSQL = String.format("DROP DATABASE %s", dbNameToReset);
logStandardOut(dropDbSQL);
databaseConnection.execStatement(dropDbSQL);
break;
} catch (MissingDatabaseException ex) {
//ignore, if the DB was not there (to be dropped)
logStandardOut(ex.getMessage());
break;
} catch (Exception ex) {
logStandardOut("Something went wrong, sleeping for a bit: %s, %s", ex.getMessage(), ex.getCause());
sleepMilliSec(1000);
}
}
GRANT CONNECT ON DATABASE thedb TO public;