Search code examples
c#.netprintingwmi

How to Determine Which Printers are Connected using WMI


I've been trying to find a way to figure out which installed printers are 'connected'. After some Googling I figured I had to dive into WMI.

So I've built this test:

// Struct to store printer data in.
public struct MyPrinter 
{ 
    public string Availability; 
    public string ExtendedPrinterStatus; 
    public string Name; 
    public string PrinterStatus; 
    public string Status; 
    public string StatusInfo; 

    public MyPrinter(string a, string eps, string n, string ps, string s, string si) 
    { 
        Availability = a; 
        ExtendedPrinterStatus = eps; 
        Name = n; 
        PrinterStatus = ps; 
        Status = s; 
        StatusInfo = si; 
    } 
}


var installedPrinters = new string[numPrinters];
PrinterSettings.InstalledPrinters.CopyTo(installedPrinters, 0);

var searcher = new ManagementObjectSearcher("SELECT * FROM Win32_Printer"); 
var data = new List<MyPrinter>(); 

foreach (var printer in searcher.Get()) 
{ 
    if (installedPrinters.Contains(printer["Name"].ToString()))
    {
        var availability = (printer["Availability"] ?? "").ToString(); 
        var extendedPrinterStatus = (printer["ExtendedPrinterStatus"] ?? "").ToString(); 
        var name = (printer["Name"] ?? "").ToString(); 
        var printerStatus = (printer["PrinterStatus"] ?? "").ToString(); 
        var status = (printer["Status"] ?? "").ToString(); 
        var statusInfo = (printer["StatusInfo"] ?? "").ToString(); 

        data.Add(new MyPrinter(availability, extendedPrinterStatus, name, printerStatus, status, statusInfo)); 
    }
}

I have 6 printers from which 2 are network printers. I've run this with all printers connected and all results looked like this:

Availability = "" // printer["Availability"] = null
ExtendedPrinterStatus = "2" // 2 = Unknown
Name = "{printer name here}"
PrinterStatus = "3" // 3 = Idle
Status = "Unknown"
StatusInfo = "" // Null

So the only difference between the printers is the name. I ran the script again but this time I disconnected my laptop from the network. So 2 of the printers were not connected anymore in this case.

The strange thing (for me) is, the results were exactly the same.

The reason I ran this test is, to figure out which field I'd need to use for my case.

So at the end, I have not been able to figure out how to figure out if a printer is connected or not. So what I'd like, is a way to figure out the installed + connected printers in C#. If there is a way to do it without the use of WMI classes, that's also fine by me, as long as it works.


Solution

  • A colleague and I have tried lots of things to find a solution for this and found that this worked:

    private string[] GetAvailablePrinters()
    {
        var installedPrinters = new string[PrinterSettings.InstalledPrinters.Count];
        PrinterSettings.InstalledPrinters.CopyTo(installedPrinters, 0);
    
        var printers = new List<string>();
        var printServers = new List<string>();
        var searcher = new ManagementObjectSearcher("SELECT * FROM Win32_Printer");
    
        foreach (var printer in searcher.Get())
        {
            var serverName = @"\\" + printer["SystemName"].ToString().TrimStart('\\');
            if (!printServers.Contains(serverName))
                printServers.Add(serverName);
        }
    
        foreach (var printServer in printServers)
        {
            var server = new PrintServer(printServer);
            try
            {
                var queues = server.GetPrintQueues();
                printers.AddRange(queues.Select(q => q.Name));
            }
            catch (Exception)
            {
                // Handle exception correctly
            }
        }
    
        return printers.ToArray();
    }
    

    The trick is that when a printserver is not available, GetPrintQueues will throw some specific exception. By only adding the printers that don't throw such an exception, we get a list of all the connected printers. This doesn't check if a printer is turned on/off because that actually doesn't matter. If it is turned off, the document will just be placed in the print queue and it can be printed later on.

    I hope this helps others who bump into this problem.

    Sidenote: The reason I decided not to catch that specific exception, is because I would have to reference a dll just for that exception.