diff --git a/Makefile.am b/Makefile.am
index 6c25dcaaaa0cdf274aa8a31a163833ecb0605351..612a98a5e55f547fe9535810c0d35f8b3e88738d 100644
--- a/Makefile.am
+++ b/Makefile.am
@@ -449,7 +449,11 @@ python_tests = \
 	test/ganeti.ht_unittest.py \
 	test/ganeti.http_unittest.py \
 	test/ganeti.hypervisor_unittest.py \
+	test/ganeti.hypervisor.hv_chroot_unittest.py \
+	test/ganeti.hypervisor.hv_fake_unittest.py \
 	test/ganeti.hypervisor.hv_kvm_unittest.py \
+	test/ganeti.hypervisor.hv_lxc_unittest.py \
+	test/ganeti.hypervisor.hv_xen_unittest.py \
 	test/ganeti.impexpd_unittest.py \
 	test/ganeti.jqueue_unittest.py \
 	test/ganeti.locking_unittest.py \
diff --git a/lib/cmdlib.py b/lib/cmdlib.py
index 996489e5cfa294a7d7b7f696f22f7c838c8b6e14..5a4a08fa59903c881cf630b7f3d0728205e4828a 100644
--- a/lib/cmdlib.py
+++ b/lib/cmdlib.py
@@ -7691,14 +7691,9 @@ class LUConnectConsole(NoHooksLU):
     # instance and then saving the defaults in the instance itself.
     hvparams = cluster.FillHV(instance)
     beparams = cluster.FillBE(instance)
-    console_cmd = hyper.GetShellCommandForConsole(instance, hvparams, beparams)
-
-    console = objects.InstanceConsole(instance=instance.name,
-                                      kind=constants.CONS_SSH,
-                                      host=node,
-                                      user="root",
-                                      command=console_cmd)
+    console = hyper.GetInstanceConsole(instance, hvparams, beparams)
 
+    assert console.instance == instance.name
     assert console.Validate()
 
     return console.ToDict()
diff --git a/lib/hypervisor/hv_base.py b/lib/hypervisor/hv_base.py
index 80675eedda5e25f6e418e0d4d9f2d0b1e264fe6c..ec474005ced6c21d4c0dbc60ea88f5b72603952e 100644
--- a/lib/hypervisor/hv_base.py
+++ b/lib/hypervisor/hv_base.py
@@ -206,8 +206,8 @@ class BaseHypervisor(object):
     raise NotImplementedError
 
   @classmethod
-  def GetShellCommandForConsole(cls, instance, hvparams, beparams):
-    """Return a command for connecting to the console of an instance.
+  def GetInstanceConsole(cls, instance, hvparams, beparams):
+    """Return information for connecting to the console of an instance.
 
     """
     raise NotImplementedError
diff --git a/lib/hypervisor/hv_chroot.py b/lib/hypervisor/hv_chroot.py
index f33fed2041319e86d5986214546087e17fcbb928..5b26e8b4441739fe08eed3e099dbc705403a0298 100644
--- a/lib/hypervisor/hv_chroot.py
+++ b/lib/hypervisor/hv_chroot.py
@@ -31,6 +31,7 @@ import logging
 from ganeti import constants
 from ganeti import errors # pylint: disable-msg=W0611
 from ganeti import utils
+from ganeti import objects
 from ganeti.hypervisor import hv_base
 from ganeti.errors import HypervisorError
 
@@ -246,15 +247,21 @@ class ChrootManager(hv_base.BaseHypervisor):
     return self.GetLinuxNodeInfo()
 
   @classmethod
-  def GetShellCommandForConsole(cls, instance, hvparams, beparams):
-    """Return a command for connecting to the console of an instance.
+  def GetInstanceConsole(cls, instance, # pylint: disable-msg=W0221
+                         hvparams, beparams, root_dir=None):
+    """Return information for connecting to the console of an instance.
 
     """
-    root_dir = cls._InstanceDir(instance.name)
-    if not os.path.ismount(root_dir):
-      raise HypervisorError("Instance %s is not running" % instance.name)
-
-    return "chroot %s" % root_dir
+    if root_dir is None:
+      root_dir = cls._InstanceDir(instance.name)
+      if not os.path.ismount(root_dir):
+        raise HypervisorError("Instance %s is not running" % instance.name)
+
+    return objects.InstanceConsole(instance=instance.name,
+                                   kind=constants.CONS_SSH,
+                                   host=instance.primary_node,
+                                   user=constants.GANETI_RUNAS,
+                                   command=["chroot", root_dir])
 
   def Verify(self):
     """Verify the hypervisor.
diff --git a/lib/hypervisor/hv_fake.py b/lib/hypervisor/hv_fake.py
index 2d386d2e3190c6bfa6aa896420c4013ad019f939..9e5f96e32c9d1650e7aac44e855bf339151e5e13 100644
--- a/lib/hypervisor/hv_fake.py
+++ b/lib/hypervisor/hv_fake.py
@@ -30,6 +30,7 @@ import logging
 from ganeti import utils
 from ganeti import constants
 from ganeti import errors
+from ganeti import objects
 from ganeti.hypervisor import hv_base
 
 
@@ -204,11 +205,14 @@ class FakeHypervisor(hv_base.BaseHypervisor):
     return result
 
   @classmethod
-  def GetShellCommandForConsole(cls, instance, hvparams, beparams):
-    """Return a command for connecting to the console of an instance.
+  def GetInstanceConsole(cls, instance, hvparams, beparams):
+    """Return information for connecting to the console of an instance.
 
     """
-    return "echo Console not available for fake hypervisor"
+    return objects.InstanceConsole(instance=instance.name,
+                                   kind=constants.CONS_MESSAGE,
+                                   message=("Console not available for fake"
+                                            " hypervisor"))
 
   def Verify(self):
     """Verify the hypervisor.
diff --git a/lib/hypervisor/hv_kvm.py b/lib/hypervisor/hv_kvm.py
index 0538ffb62536dbe1232e28ab2b1f29f9f8435cd1..46f41c5b47139190c0586ef44222c86393d5484c 100644
--- a/lib/hypervisor/hv_kvm.py
+++ b/lib/hypervisor/hv_kvm.py
@@ -1035,28 +1035,33 @@ class KVMHypervisor(hv_base.BaseHypervisor):
     return self.GetLinuxNodeInfo()
 
   @classmethod
-  def GetShellCommandForConsole(cls, instance, hvparams, beparams):
+  def GetInstanceConsole(cls, instance, hvparams, beparams):
     """Return a command for connecting to the console of an instance.
 
     """
     if hvparams[constants.HV_SERIAL_CONSOLE]:
-      shell_command = ("%s STDIO,%s UNIX-CONNECT:%s" %
-                       (constants.SOCAT_PATH, cls._SocatUnixConsoleParams(),
-                        utils.ShellQuote(cls._InstanceSerial(instance.name))))
-    else:
-      shell_command = "echo 'No serial shell for instance %s'" % instance.name
+      cmd = [constants.SOCAT_PATH,
+             "STDIO,%s" % cls._SocatUnixConsoleParams(),
+             "UNIX-CONNECT:%s" % cls._InstanceSerial(instance.name)]
+      return objects.InstanceConsole(instance=instance.name,
+                                     kind=constants.CONS_SSH,
+                                     host=instance.primary_node,
+                                     user=constants.GANETI_RUNAS,
+                                     command=cmd)
 
     vnc_bind_address = hvparams[constants.HV_VNC_BIND_ADDRESS]
-    if vnc_bind_address:
-      if instance.network_port > constants.VNC_BASE_PORT:
-        display = instance.network_port - constants.VNC_BASE_PORT
-        vnc_command = ("echo 'Instance has VNC listening on %s:%d"
-                       " (display: %d)'" % (vnc_bind_address,
-                                            instance.network_port,
-                                            display))
-        shell_command = "%s; %s" % (vnc_command, shell_command)
-
-    return shell_command
+    if vnc_bind_address and instance.network_port > constants.VNC_BASE_PORT:
+      display = instance.network_port - constants.VNC_BASE_PORT
+      return objects.InstanceConsole(instance=instance.name,
+                                     kind=constants.CONS_VNC,
+                                     host=vnc_bind_address,
+                                     port=instance.network_port,
+                                     display=display)
+
+    return objects.InstanceConsole(instance=instance.name,
+                                   kind=constants.CONS_MESSAGE,
+                                   message=("No serial shell for instance %s" %
+                                            instance.name))
 
   def Verify(self):
     """Verify the hypervisor.
@@ -1069,7 +1074,6 @@ class KVMHypervisor(hv_base.BaseHypervisor):
     if not os.path.exists(constants.SOCAT_PATH):
       return "The socat binary ('%s') does not exist." % constants.SOCAT_PATH
 
-
   @classmethod
   def CheckParameterSyntax(cls, hvparams):
     """Check the given parameters for validity.
diff --git a/lib/hypervisor/hv_lxc.py b/lib/hypervisor/hv_lxc.py
index 3779a68cd6a077293c3d629c3f59142386e7d599..78c1f3467b611a881f0989200920f4fd14a2e20f 100644
--- a/lib/hypervisor/hv_lxc.py
+++ b/lib/hypervisor/hv_lxc.py
@@ -31,6 +31,7 @@ import logging
 from ganeti import constants
 from ganeti import errors # pylint: disable-msg=W0611
 from ganeti import utils
+from ganeti import objects
 from ganeti.hypervisor import hv_base
 from ganeti.errors import HypervisorError
 
@@ -374,11 +375,15 @@ class LXCHypervisor(hv_base.BaseHypervisor):
     return self.GetLinuxNodeInfo()
 
   @classmethod
-  def GetShellCommandForConsole(cls, instance, hvparams, beparams):
+  def GetInstanceConsole(cls, instance, hvparams, beparams):
     """Return a command for connecting to the console of an instance.
 
     """
-    return "lxc-console -n %s" % instance.name
+    return objects.InstanceConsole(instance=instance.name,
+                                   kind=constants.CONS_SSH,
+                                   host=instance.primary_node,
+                                   user=constants.GANETI_RUNAS,
+                                   command=["lxc-console", "-n", instance.name])
 
   def Verify(self):
     """Verify the hypervisor.
diff --git a/lib/hypervisor/hv_xen.py b/lib/hypervisor/hv_xen.py
index 5d6de5cc8c0456e4442fc422c8c39213db0ca48b..8d36e202a10cda3471d4f6d58c9c6b2a5c825301 100644
--- a/lib/hypervisor/hv_xen.py
+++ b/lib/hypervisor/hv_xen.py
@@ -31,6 +31,7 @@ from ganeti import errors
 from ganeti import utils
 from ganeti.hypervisor import hv_base
 from ganeti import netutils
+from ganeti import objects
 
 
 class XenHypervisor(hv_base.BaseHypervisor):
@@ -294,12 +295,15 @@ class XenHypervisor(hv_base.BaseHypervisor):
     return result
 
   @classmethod
-  def GetShellCommandForConsole(cls, instance, hvparams, beparams):
+  def GetInstanceConsole(cls, instance, hvparams, beparams):
     """Return a command for connecting to the console of an instance.
 
     """
-    return "xm console %s" % instance.name
-
+    return objects.InstanceConsole(instance=instance.name,
+                                   kind=constants.CONS_SSH,
+                                   host=instance.primary_node,
+                                   user=constants.GANETI_RUNAS,
+                                   command=["xm", "console", instance.name])
 
   def Verify(self):
     """Verify the hypervisor.
diff --git a/test/ganeti.hypervisor.hv_chroot_unittest.py b/test/ganeti.hypervisor.hv_chroot_unittest.py
new file mode 100755
index 0000000000000000000000000000000000000000..233808c4acd7eb778cf7ccdab614aae7fd19bfe9
--- /dev/null
+++ b/test/ganeti.hypervisor.hv_chroot_unittest.py
@@ -0,0 +1,54 @@
+#!/usr/bin/python
+#
+
+# Copyright (C) 2011 Google Inc.
+#
+# This program is free software; you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation; either version 2 of the License, or
+# (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful, but
+# WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+# General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
+# 02110-1301, USA.
+
+
+"""Script for testing ganeti.hypervisor.hv_chroot"""
+
+import unittest
+import tempfile
+import shutil
+
+from ganeti import constants
+from ganeti import objects
+from ganeti import hypervisor
+
+from ganeti.hypervisor import hv_chroot
+
+import testutils
+
+
+class TestConsole(unittest.TestCase):
+  def setUp(self):
+    self.tmpdir = tempfile.mkdtemp()
+
+  def tearDown(self):
+    shutil.rmtree(self.tmpdir)
+
+  def test(self):
+    instance = objects.Instance(name="fake.example.com", primary_node="node837")
+    cons = hv_chroot.ChrootManager.GetInstanceConsole(instance, {}, {},
+                                                      root_dir=self.tmpdir)
+    self.assertTrue(cons.Validate())
+    self.assertEqual(cons.kind, constants.CONS_SSH)
+    self.assertEqual(cons.host, instance.primary_node)
+
+
+if __name__ == "__main__":
+  testutils.GanetiTestProgram()
diff --git a/test/ganeti.hypervisor.hv_fake_unittest.py b/test/ganeti.hypervisor.hv_fake_unittest.py
new file mode 100755
index 0000000000000000000000000000000000000000..e0c42407bbff2a0b0f2d44e900c9b5b4716b679d
--- /dev/null
+++ b/test/ganeti.hypervisor.hv_fake_unittest.py
@@ -0,0 +1,44 @@
+#!/usr/bin/python
+#
+
+# Copyright (C) 2011 Google Inc.
+#
+# This program is free software; you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation; either version 2 of the License, or
+# (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful, but
+# WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+# General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
+# 02110-1301, USA.
+
+
+"""Script for testing ganeti.hypervisor.hv_fake"""
+
+import unittest
+
+from ganeti import constants
+from ganeti import objects
+from ganeti import hypervisor
+
+from ganeti.hypervisor import hv_fake
+
+import testutils
+
+
+class TestConsole(unittest.TestCase):
+  def test(self):
+    instance = objects.Instance(name="fake.example.com")
+    cons = hv_fake.FakeHypervisor.GetInstanceConsole(instance, {}, {})
+    self.assertTrue(cons.Validate())
+    self.assertEqual(cons.kind, constants.CONS_MESSAGE)
+
+
+if __name__ == "__main__":
+  testutils.GanetiTestProgram()
diff --git a/test/ganeti.hypervisor.hv_kvm_unittest.py b/test/ganeti.hypervisor.hv_kvm_unittest.py
index 28a936be1dbb9c9b8d66d9a49931fabf580761f9..c2a964f7c9d378400f8945a8d3ac293b2682a5d2 100755
--- a/test/ganeti.hypervisor.hv_kvm_unittest.py
+++ b/test/ganeti.hypervisor.hv_kvm_unittest.py
@@ -1,7 +1,7 @@
 #!/usr/bin/python
 #
 
-# Copyright (C) 2010 Google Inc.
+# Copyright (C) 2010, 2011 Google Inc.
 #
 # This program is free software; you can redistribute it and/or modify
 # it under the terms of the GNU General Public License as published by
@@ -80,5 +80,49 @@ class TestWriteNetScript(unittest.TestCase):
                       inst, nic, 2)
 
 
+class TestConsole(unittest.TestCase):
+  def _Test(self, instance, hvparams):
+    cons = hv_kvm.KVMHypervisor.GetInstanceConsole(instance, hvparams, {})
+    self.assertTrue(cons.Validate())
+    return cons
+
+  def testSerial(self):
+    instance = objects.Instance(name="kvm.example.com",
+                                primary_node="node6017")
+    hvparams = {
+      constants.HV_SERIAL_CONSOLE: True,
+      constants.HV_VNC_BIND_ADDRESS: None,
+      }
+    cons = self._Test(instance, hvparams)
+    self.assertEqual(cons.kind, constants.CONS_SSH)
+    self.assertEqual(cons.host, instance.primary_node)
+    self.assertEqual(cons.command[0], constants.SOCAT_PATH)
+
+  def testVnc(self):
+    instance = objects.Instance(name="kvm.example.com",
+                                primary_node="node7235",
+                                network_port=constants.VNC_BASE_PORT + 10)
+    hvparams = {
+      constants.HV_SERIAL_CONSOLE: False,
+      constants.HV_VNC_BIND_ADDRESS: "192.0.2.1",
+      }
+    cons = self._Test(instance, hvparams)
+    self.assertEqual(cons.kind, constants.CONS_VNC)
+    self.assertEqual(cons.host, "192.0.2.1")
+    self.assertEqual(cons.port, constants.VNC_BASE_PORT + 10)
+    self.assertEqual(cons.display, 10)
+
+  def testNoConsole(self):
+    instance = objects.Instance(name="kvm.example.com",
+                                primary_node="node24325",
+                                network_port=0)
+    hvparams = {
+      constants.HV_SERIAL_CONSOLE: False,
+      constants.HV_VNC_BIND_ADDRESS: None,
+      }
+    cons = self._Test(instance, hvparams)
+    self.assertEqual(cons.kind, constants.CONS_MESSAGE)
+
+
 if __name__ == "__main__":
   testutils.GanetiTestProgram()
diff --git a/test/ganeti.hypervisor.hv_lxc_unittest.py b/test/ganeti.hypervisor.hv_lxc_unittest.py
new file mode 100755
index 0000000000000000000000000000000000000000..9d9e441ed08ade54abb643ef728dd35335bcc438
--- /dev/null
+++ b/test/ganeti.hypervisor.hv_lxc_unittest.py
@@ -0,0 +1,46 @@
+#!/usr/bin/python
+#
+
+# Copyright (C) 2011 Google Inc.
+#
+# This program is free software; you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation; either version 2 of the License, or
+# (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful, but
+# WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+# General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
+# 02110-1301, USA.
+
+
+"""Script for testing ganeti.hypervisor.hv_lxc"""
+
+import unittest
+
+from ganeti import constants
+from ganeti import objects
+from ganeti import hypervisor
+
+from ganeti.hypervisor import hv_lxc
+
+import testutils
+
+
+class TestConsole(unittest.TestCase):
+  def test(self):
+    instance = objects.Instance(name="lxc.example.com", primary_node="node199")
+    cons = hv_lxc.LXCHypervisor.GetInstanceConsole(instance, {}, {})
+    self.assertTrue(cons.Validate())
+    self.assertEqual(cons.kind, constants.CONS_SSH)
+    self.assertEqual(cons.host, instance.primary_node)
+    self.assertEqual(cons.command[-1], instance.name)
+
+
+if __name__ == "__main__":
+  testutils.GanetiTestProgram()
diff --git a/test/ganeti.hypervisor.hv_xen_unittest.py b/test/ganeti.hypervisor.hv_xen_unittest.py
new file mode 100755
index 0000000000000000000000000000000000000000..dc868396e6b5a71941c283a9efb605e0bed65e78
--- /dev/null
+++ b/test/ganeti.hypervisor.hv_xen_unittest.py
@@ -0,0 +1,48 @@
+#!/usr/bin/python
+#
+
+# Copyright (C) 2011 Google Inc.
+#
+# This program is free software; you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation; either version 2 of the License, or
+# (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful, but
+# WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+# General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
+# 02110-1301, USA.
+
+
+"""Script for testing ganeti.hypervisor.hv_lxc"""
+
+import unittest
+
+from ganeti import constants
+from ganeti import objects
+from ganeti import hypervisor
+
+from ganeti.hypervisor import hv_xen
+
+import testutils
+
+
+class TestConsole(unittest.TestCase):
+  def test(self):
+    for cls in [hv_xen.XenPvmHypervisor, hv_xen.XenHvmHypervisor]:
+      instance = objects.Instance(name="xen.example.com",
+                                  primary_node="node24828")
+      cons = cls.GetInstanceConsole(instance, {}, {})
+      self.assertTrue(cons.Validate())
+      self.assertEqual(cons.kind, constants.CONS_SSH)
+      self.assertEqual(cons.host, instance.primary_node)
+      self.assertEqual(cons.command[-1], instance.name)
+
+
+if __name__ == "__main__":
+  testutils.GanetiTestProgram()