Help! My database is stuck in SINGLE_USER mode!

Help!

HELP! My database is stuck in single user mode!

I answered this twitter post a while back and figured it would make a fun blog post.

So what do I mean that the database is stuck in single user mode? Simply speaking, it means that the database is in single user mode and you can’t seem to get it back into multi user mode. As an example I set my local AdventureWorks2012 database to single user mode, opened a session to that database, tried an alter database to get it back to multi user and this is what I got.


I’m stuck!

Yup! Most definitely stuck. You can see from the error message that it won’t go back to MULTI_USER!

I know I have gotten myself into this situation more than once. For whatever reason I thought it would be a good idea to put the database in single user mode, and depending on database activity, it caused me a minor headache to get it back out. If it’s the first time you’ve ever done it, it can definitely be one of those “oh crap” moments. You know the one. Where all the blood drains out of your face, you start to sweat and you wonder if you’ll still have a job in the morning. Yeah, that one.

I’m here to tell you that your predicament is not that dire. It usually it means that there is another session to that database preventing your session from performing your alter database, and the most typical resolution is to find the session that has control of your database and kill it!

Here is the connection that has control of my AdventureWorks2012 database.

I’m stuck!

This is the session holding the one connection to my AdventureWorks2012 database.

Here is the TSQL used to find the session that you need to kill.

USE [master] ;
 
DECLARE @DatabaseName VARCHAR(255) ;
SET @DatabaseName = 'AdventureWorks2012' ;
 
SELECT spr.spid AS [spid]
     , sdb.NAME AS [DatabaseName]
     , spr.open_tran AS [OpenTransactions]
     , spr.status AS [Status]
     , ('kill ' + CAST(spr.spid AS VARCHAR)) AS [KillCommand]
FROM sys.sysprocesses spr
INNER JOIN sys.databases sdb ON sdb.database_id = spr.dbid
WHERE sdb.NAME = @DatabaseName ;

And here are the results from the TSQL.

I’m stuck!

Here are the results from the session find script, and you can see that is calls out SPID 53.

So at this point I know I have to kill SPID 53 so I can proceed with getting my database back into multi user mode. I don’t like having my script automatically kill it in case there are open transactions. Yes it adds another step to my process, but if there is a legitimate transaction open I may want to consider what it is doing before I go killing it.

As you can see SPID 53 has no open transactions so I determine that I am OK to kill it at this point.


I’m stuck!

And as you can see after killing SPID 53 I was able to set it back into multi user mode.

And here we are back in business! I have never had this not work. I have had some issues with a very persistent application trying to connect before I was able to complete the process of finding the spid, killing it and altering my database, but it has always worked for me. @sqL_handLe did mention that Async Statistics could also possibly consume a connection, and that might not show up in the session results using the query above, but I have not seen this problem so I cannot speak from experience on that. Just another good to know tidbit.

So if you end up getting your database stuck in single user mode, don’t worry, there’s usually a way out.


See also