2013-03-17 17:06:52 -04:00
#
2013-10-27 21:59:46 +01:00
# Copyright 2009, 2013 Red Hat, Inc.
2013-03-17 17:06:52 -04:00
# Cole Robinson <crobinso@redhat.com>
#
2018-04-04 14:35:41 +01:00
# This work is licensed under the GNU GPLv2 or later.
2018-03-20 15:00:02 -04:00
# See the COPYING file in the top-level directory.
2013-03-17 17:06:52 -04:00
2018-03-20 12:18:35 -04:00
from . device import Device
2018-03-20 12:27:37 -04:00
from . . xmlbuilder import XMLProperty
2013-03-17 17:06:52 -04:00
2013-04-13 14:34:52 -04:00
2018-03-20 12:18:35 -04:00
class _DeviceChar ( Device ) :
2013-03-17 17:06:52 -04:00
"""
Base class for all character devices . Shouldn ' t be instantiated
directly .
"""
2013-07-16 09:14:37 -04:00
TYPE_PTY = " pty "
TYPE_DEV = " dev "
TYPE_STDIO = " stdio "
TYPE_PIPE = " pipe "
TYPE_FILE = " file "
TYPE_VC = " vc "
TYPE_NULL = " null "
TYPE_TCP = " tcp "
TYPE_UDP = " udp "
TYPE_UNIX = " unix "
TYPE_SPICEVMC = " spicevmc "
2014-03-25 15:42:33 +01:00
TYPE_SPICEPORT = " spiceport "
2016-08-24 16:37:36 -04:00
TYPE_NMDM = " nmdm "
2014-03-25 15:42:33 +01:00
2013-10-05 11:07:32 -04:00
# We don't list the non-UI friendly types here
_TYPES_FOR_ALL = [ TYPE_PTY , TYPE_DEV , TYPE_FILE ,
TYPE_TCP , TYPE_UDP , TYPE_UNIX ]
2014-03-25 15:42:33 +01:00
_TYPES_FOR_CHANNEL = [ TYPE_SPICEVMC , TYPE_SPICEPORT ]
2013-08-07 19:04:18 -04:00
TYPES = _TYPES_FOR_ALL
2013-07-16 09:14:37 -04:00
MODE_CONNECT = " connect "
MODE_BIND = " bind "
MODES = [ MODE_CONNECT , MODE_BIND ]
2013-10-05 13:27:11 -04:00
CHANNEL_NAME_SPICE = " com.redhat.spice.0 "
CHANNEL_NAME_QEMUGA = " org.qemu.guest_agent.0 "
CHANNEL_NAME_LIBGUESTFS = " org.libguestfs.channel.0 "
2014-03-25 15:42:33 +01:00
CHANNEL_NAME_SPICE_WEBDAV = " org.spice-space.webdav.0 "
2013-10-05 13:27:11 -04:00
CHANNEL_NAMES = [ CHANNEL_NAME_SPICE ,
CHANNEL_NAME_QEMUGA ,
2014-03-25 15:42:33 +01:00
CHANNEL_NAME_LIBGUESTFS ,
CHANNEL_NAME_SPICE_WEBDAV ]
2013-10-05 13:27:11 -04:00
2013-10-05 14:04:49 -04:00
@staticmethod
def pretty_channel_name ( val ) :
2018-03-20 12:18:35 -04:00
if val == _DeviceChar . CHANNEL_NAME_SPICE :
2013-10-05 14:04:49 -04:00
return " spice "
2018-03-20 12:18:35 -04:00
if val == _DeviceChar . CHANNEL_NAME_QEMUGA :
2013-10-05 14:04:49 -04:00
return " qemu-ga "
2018-03-20 12:18:35 -04:00
if val == _DeviceChar . CHANNEL_NAME_LIBGUESTFS :
2013-10-05 14:04:49 -04:00
return " libguestfs "
2018-03-20 12:18:35 -04:00
if val == _DeviceChar . CHANNEL_NAME_SPICE_WEBDAV :
2014-03-25 15:42:33 +01:00
return " spice-webdav "
2013-10-05 14:04:49 -04:00
return None
2013-07-16 09:14:37 -04:00
@staticmethod
def pretty_type ( ctype ) :
2013-03-17 17:06:52 -04:00
"""
Return a human readable description of the passed char type
"""
desc = " "
2018-03-20 12:18:35 -04:00
if ctype == _DeviceChar . TYPE_PTY :
2013-03-17 17:06:52 -04:00
desc = _ ( " Pseudo TTY " )
2018-03-20 12:18:35 -04:00
elif ctype == _DeviceChar . TYPE_DEV :
2013-03-17 17:06:52 -04:00
desc = _ ( " Physical host character device " )
2018-03-20 12:18:35 -04:00
elif ctype == _DeviceChar . TYPE_STDIO :
2013-03-17 17:06:52 -04:00
desc = _ ( " Standard input/output " )
2018-03-20 12:18:35 -04:00
elif ctype == _DeviceChar . TYPE_PIPE :
2013-03-17 17:06:52 -04:00
desc = _ ( " Named pipe " )
2018-03-20 12:18:35 -04:00
elif ctype == _DeviceChar . TYPE_FILE :
2013-03-17 17:06:52 -04:00
desc = _ ( " Output to a file " )
2018-03-20 12:18:35 -04:00
elif ctype == _DeviceChar . TYPE_VC :
2013-03-17 17:06:52 -04:00
desc = _ ( " Virtual console " )
2018-03-20 12:18:35 -04:00
elif ctype == _DeviceChar . TYPE_NULL :
2013-03-17 17:06:52 -04:00
desc = _ ( " Null device " )
2018-03-20 12:18:35 -04:00
elif ctype == _DeviceChar . TYPE_TCP :
2013-03-17 17:06:52 -04:00
desc = _ ( " TCP net console " )
2018-03-20 12:18:35 -04:00
elif ctype == _DeviceChar . TYPE_UDP :
2013-03-17 17:06:52 -04:00
desc = _ ( " UDP net console " )
2018-03-20 12:18:35 -04:00
elif ctype == _DeviceChar . TYPE_UNIX :
2013-03-17 17:06:52 -04:00
desc = _ ( " Unix socket " )
2018-03-20 12:18:35 -04:00
elif ctype == _DeviceChar . TYPE_SPICEVMC :
2013-03-17 17:06:52 -04:00
desc = _ ( " Spice agent " )
2018-03-20 12:18:35 -04:00
elif ctype == _DeviceChar . TYPE_SPICEPORT :
2014-03-25 15:42:33 +01:00
desc = _ ( " Spice port " )
2013-03-17 17:06:52 -04:00
return desc
2013-07-16 09:14:37 -04:00
@staticmethod
def pretty_mode ( char_mode ) :
2013-03-17 17:06:52 -04:00
"""
Return a human readable description of the passed char type
"""
desc = " "
2018-03-20 12:18:35 -04:00
if char_mode == _DeviceChar . MODE_CONNECT :
2013-03-17 17:06:52 -04:00
desc = _ ( " Client mode " )
2018-03-20 12:18:35 -04:00
elif char_mode == _DeviceChar . MODE_BIND :
2013-03-17 17:06:52 -04:00
desc = _ ( " Server mode " )
return desc
def supports_property ( self , propname , ro = False ) :
"""
Whether the character dev type supports the passed property name
"""
users = {
2017-08-05 14:39:32 +08:00
" source_path " : [ self . TYPE_FILE , self . TYPE_UNIX ,
self . TYPE_DEV , self . TYPE_PIPE ] ,
" source_mode " : [ self . TYPE_UNIX , self . TYPE_TCP ] ,
" source_host " : [ self . TYPE_TCP , self . TYPE_UDP ] ,
" source_port " : [ self . TYPE_TCP , self . TYPE_UDP ] ,
" source_channel " : [ self . TYPE_SPICEPORT ] ,
" source_master " : [ self . TYPE_NMDM ] ,
" source_slave " : [ self . TYPE_NMDM ] ,
" protocol " : [ self . TYPE_TCP ] ,
" bind_host " : [ self . TYPE_UDP ] ,
" bind_port " : [ self . TYPE_UDP ] ,
2013-10-05 11:28:41 -04:00
}
2013-03-17 17:06:52 -04:00
if ro :
2013-07-16 09:14:37 -04:00
users [ " source_path " ] + = [ self . TYPE_PTY ]
2013-03-17 17:06:52 -04:00
if users . get ( propname ) :
2013-07-16 09:14:37 -04:00
return self . type in users [ propname ]
2013-03-17 17:06:52 -04:00
return hasattr ( self , propname )
2014-01-19 13:56:06 -05:00
def _set_host_helper ( self , hostparam , portparam , val ) :
def parse_host ( val ) :
host , ignore , port = ( val or " " ) . partition ( " : " )
return host or None , port or None
host , port = parse_host ( val )
2015-09-05 16:27:27 -04:00
if not host :
host = " 127.0.0.1 "
2014-01-19 13:56:06 -05:00
if host :
setattr ( self , hostparam , host )
if port :
setattr ( self , portparam , port )
def set_friendly_source ( self , val ) :
self . _set_host_helper ( " source_host " , " source_port " , val )
def set_friendly_bind ( self , val ) :
self . _set_host_helper ( " bind_host " , " bind_port " , val )
def set_friendly_target ( self , val ) :
self . _set_host_helper ( " target_address " , " target_port " , val )
2013-07-16 09:14:37 -04:00
2018-09-03 16:18:03 -04:00
_XML_PROP_ORDER = [ " type " ,
2013-07-16 09:14:37 -04:00
" bind_host " , " bind_port " ,
2015-09-05 16:27:27 -04:00
" source_mode " , " source_host " , " source_port " ,
" _source_path " , " source_channel " ,
2018-07-04 16:10:38 +08:00
" target_type " , " target_name " , " target_state " ]
2013-07-16 09:14:37 -04:00
2018-02-22 20:44:09 -05:00
type = XMLProperty ( " ./@type " )
2013-09-24 09:25:05 -04:00
_tty = XMLProperty ( " ./@tty " )
2018-02-22 20:44:09 -05:00
_source_path = XMLProperty ( " ./source/@path " )
2013-09-24 09:25:05 -04:00
def _get_source_path ( self ) :
source = self . _source_path
if source is None and self . _tty :
return self . _tty
return source
def _set_source_path ( self , val ) :
self . _source_path = val
source_path = property ( _get_source_path , _set_source_path )
2013-07-16 09:14:37 -04:00
2018-02-22 20:44:09 -05:00
source_channel = XMLProperty ( " ./source/@channel " )
2016-08-24 16:37:36 -04:00
source_master = XMLProperty ( " ./source/@master " )
source_slave = XMLProperty ( " ./source/@slave " )
2014-03-25 15:42:32 +01:00
2018-07-04 16:10:38 +08:00
target_state = XMLProperty ( " ./target/@state " )
2015-05-06 13:54:00 -04:00
2015-09-05 16:27:27 -04:00
###################
# source handling #
###################
2015-05-06 13:54:00 -04:00
2015-09-05 16:27:27 -04:00
source_mode = XMLProperty ( " ./source/@mode " )
2013-07-16 09:14:37 -04:00
2018-09-03 16:18:03 -04:00
source_host = XMLProperty ( " ./source[@mode= ' connect ' ]/@host " )
source_port = XMLProperty (
" ./source[@mode= ' connect ' ]/@service " , is_int = True )
bind_host = XMLProperty ( " ./source[@mode= ' bind ' ]/@host " )
bind_port = XMLProperty ( " ./source[@mode= ' bind ' ]/@service " , is_int = True )
2015-09-05 16:27:27 -04:00
#######################
# Remaining XML props #
#######################
2013-07-16 09:14:37 -04:00
2018-09-01 17:37:26 -04:00
protocol = XMLProperty ( " ./protocol/@type " )
2013-07-16 09:14:37 -04:00
2018-02-22 20:44:09 -05:00
target_address = XMLProperty ( " ./target/@address " )
target_port = XMLProperty ( " ./target/@port " , is_int = True )
2018-09-01 17:37:26 -04:00
target_type = XMLProperty ( " ./target/@type " )
target_name = XMLProperty ( " ./target/@name " )
2013-04-13 14:34:52 -04:00
2016-07-18 15:03:06 -04:00
log_file = XMLProperty ( " ./log/@file " )
log_append = XMLProperty ( " ./log/@append " , is_onoff = True )
2013-04-13 14:34:52 -04:00
2018-09-03 16:18:03 -04:00
##################
# Default config #
##################
def set_defaults ( self , _guest ) :
if not self . source_mode and self . supports_property ( " source_mode " ) :
self . source_mode = self . MODE_BIND
if not self . protocol and self . supports_property ( " protocol " ) :
2018-09-06 11:49:09 -04:00
self . protocol = " raw "
2018-09-03 16:18:03 -04:00
if not self . target_type and self . DEVICE_TYPE == " channel " :
2018-09-06 11:49:09 -04:00
self . target_type = " virtio "
2018-09-03 16:18:03 -04:00
if not self . target_name and self . type == self . TYPE_SPICEVMC :
self . target_name = self . CHANNEL_NAME_SPICE
2018-03-20 12:18:35 -04:00
class DeviceConsole ( _DeviceChar ) :
2018-03-21 10:53:34 -04:00
XML_NAME = " console "
2018-03-20 12:18:35 -04:00
TYPES = [ _DeviceChar . TYPE_PTY ]
2013-04-13 14:34:52 -04:00
2018-03-20 12:18:35 -04:00
class DeviceSerial ( _DeviceChar ) :
2018-03-21 10:53:34 -04:00
XML_NAME = " serial "
2013-03-17 17:06:52 -04:00
2013-04-13 14:34:52 -04:00
2018-03-20 12:18:35 -04:00
class DeviceParallel ( _DeviceChar ) :
2018-03-21 10:53:34 -04:00
XML_NAME = " parallel "
2013-04-13 14:34:52 -04:00
2018-03-20 12:18:35 -04:00
class DeviceChannel ( _DeviceChar ) :
2018-03-21 10:53:34 -04:00
XML_NAME = " channel "
2018-03-20 12:18:35 -04:00
TYPES = ( _DeviceChar . _TYPES_FOR_CHANNEL +
_DeviceChar . _TYPES_FOR_ALL )