2013-03-18 01:06:52 +04:00
#
2013-10-28 00:59:46 +04:00
# Copyright 2009, 2013 Red Hat, Inc.
2013-03-18 01:06:52 +04:00
# Cole Robinson <crobinso@redhat.com>
#
2018-03-20 22:00:02 +03:00
# This work is licensed under the GNU GPLv2.
# See the COPYING file in the top-level directory.
2013-03-18 01:06:52 +04:00
2018-03-20 19:18:35 +03:00
from . device import Device
2018-03-20 19:27:37 +03:00
from . . xmlbuilder import XMLProperty
2013-03-18 01:06:52 +04:00
2013-04-13 22:34:52 +04:00
2018-03-20 19:18:35 +03:00
class _DeviceChar ( Device ) :
2013-03-18 01:06:52 +04:00
"""
Base class for all character devices . Shouldn ' t be instantiated
directly .
"""
2013-07-16 17: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 18:42:33 +04:00
TYPE_SPICEPORT = " spiceport "
2016-08-24 23:37:36 +03:00
TYPE_NMDM = " nmdm "
2014-03-25 18:42:33 +04:00
2013-10-05 19: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 18:42:33 +04:00
_TYPES_FOR_CHANNEL = [ TYPE_SPICEVMC , TYPE_SPICEPORT ]
2013-08-08 03:04:18 +04:00
TYPES = _TYPES_FOR_ALL
2013-07-16 17:14:37 +04:00
MODE_CONNECT = " connect "
MODE_BIND = " bind "
MODES = [ MODE_CONNECT , MODE_BIND ]
PROTOCOL_RAW = " raw "
PROTOCOL_TELNET = " telnet "
PROTOCOLS = [ PROTOCOL_RAW , PROTOCOL_TELNET ]
CHANNEL_TARGET_GUESTFWD = " guestfwd "
CHANNEL_TARGET_VIRTIO = " virtio "
CHANNEL_TARGETS = [ CHANNEL_TARGET_GUESTFWD ,
CHANNEL_TARGET_VIRTIO ]
CONSOLE_TARGET_SERIAL = " serial "
CONSOLE_TARGET_UML = " uml "
CONSOLE_TARGET_XEN = " xen "
CONSOLE_TARGET_VIRTIO = " virtio "
CONSOLE_TARGETS = [ CONSOLE_TARGET_SERIAL , CONSOLE_TARGET_UML ,
CONSOLE_TARGET_XEN , CONSOLE_TARGET_VIRTIO ]
2013-10-05 21: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 18:42:33 +04:00
CHANNEL_NAME_SPICE_WEBDAV = " org.spice-space.webdav.0 "
2013-10-05 21:27:11 +04:00
CHANNEL_NAMES = [ CHANNEL_NAME_SPICE ,
CHANNEL_NAME_QEMUGA ,
2014-03-25 18:42:33 +04:00
CHANNEL_NAME_LIBGUESTFS ,
CHANNEL_NAME_SPICE_WEBDAV ]
2013-10-05 21:27:11 +04:00
2013-10-05 22:04:49 +04:00
@staticmethod
def pretty_channel_name ( val ) :
2018-03-20 19:18:35 +03:00
if val == _DeviceChar . CHANNEL_NAME_SPICE :
2013-10-05 22:04:49 +04:00
return " spice "
2018-03-20 19:18:35 +03:00
if val == _DeviceChar . CHANNEL_NAME_QEMUGA :
2013-10-05 22:04:49 +04:00
return " qemu-ga "
2018-03-20 19:18:35 +03:00
if val == _DeviceChar . CHANNEL_NAME_LIBGUESTFS :
2013-10-05 22:04:49 +04:00
return " libguestfs "
2018-03-20 19:18:35 +03:00
if val == _DeviceChar . CHANNEL_NAME_SPICE_WEBDAV :
2014-03-25 18:42:33 +04:00
return " spice-webdav "
2013-10-05 22:04:49 +04:00
return None
2013-07-16 17:14:37 +04:00
@staticmethod
def pretty_type ( ctype ) :
2013-03-18 01:06:52 +04:00
"""
Return a human readable description of the passed char type
"""
desc = " "
2018-03-20 19:18:35 +03:00
if ctype == _DeviceChar . TYPE_PTY :
2013-03-18 01:06:52 +04:00
desc = _ ( " Pseudo TTY " )
2018-03-20 19:18:35 +03:00
elif ctype == _DeviceChar . TYPE_DEV :
2013-03-18 01:06:52 +04:00
desc = _ ( " Physical host character device " )
2018-03-20 19:18:35 +03:00
elif ctype == _DeviceChar . TYPE_STDIO :
2013-03-18 01:06:52 +04:00
desc = _ ( " Standard input/output " )
2018-03-20 19:18:35 +03:00
elif ctype == _DeviceChar . TYPE_PIPE :
2013-03-18 01:06:52 +04:00
desc = _ ( " Named pipe " )
2018-03-20 19:18:35 +03:00
elif ctype == _DeviceChar . TYPE_FILE :
2013-03-18 01:06:52 +04:00
desc = _ ( " Output to a file " )
2018-03-20 19:18:35 +03:00
elif ctype == _DeviceChar . TYPE_VC :
2013-03-18 01:06:52 +04:00
desc = _ ( " Virtual console " )
2018-03-20 19:18:35 +03:00
elif ctype == _DeviceChar . TYPE_NULL :
2013-03-18 01:06:52 +04:00
desc = _ ( " Null device " )
2018-03-20 19:18:35 +03:00
elif ctype == _DeviceChar . TYPE_TCP :
2013-03-18 01:06:52 +04:00
desc = _ ( " TCP net console " )
2018-03-20 19:18:35 +03:00
elif ctype == _DeviceChar . TYPE_UDP :
2013-03-18 01:06:52 +04:00
desc = _ ( " UDP net console " )
2018-03-20 19:18:35 +03:00
elif ctype == _DeviceChar . TYPE_UNIX :
2013-03-18 01:06:52 +04:00
desc = _ ( " Unix socket " )
2018-03-20 19:18:35 +03:00
elif ctype == _DeviceChar . TYPE_SPICEVMC :
2013-03-18 01:06:52 +04:00
desc = _ ( " Spice agent " )
2018-03-20 19:18:35 +03:00
elif ctype == _DeviceChar . TYPE_SPICEPORT :
2014-03-25 18:42:33 +04:00
desc = _ ( " Spice port " )
2013-03-18 01:06:52 +04:00
return desc
2013-07-16 17:14:37 +04:00
@staticmethod
def pretty_mode ( char_mode ) :
2013-03-18 01:06:52 +04:00
"""
Return a human readable description of the passed char type
"""
desc = " "
2018-03-20 19:18:35 +03:00
if char_mode == _DeviceChar . MODE_CONNECT :
2013-03-18 01:06:52 +04:00
desc = _ ( " Client mode " )
2018-03-20 19:18:35 +03:00
elif char_mode == _DeviceChar . MODE_BIND :
2013-03-18 01: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 09:39:32 +03: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 19:28:41 +04:00
}
2013-03-18 01:06:52 +04:00
if ro :
2013-07-16 17:14:37 +04:00
users [ " source_path " ] + = [ self . TYPE_PTY ]
2013-03-18 01:06:52 +04:00
if users . get ( propname ) :
2013-07-16 17:14:37 +04:00
return self . type in users [ propname ]
2013-03-18 01:06:52 +04:00
return hasattr ( self , propname )
2015-05-06 20:54:00 +03:00
def set_defaults ( self , guest ) :
2015-09-05 23:27:27 +03:00
ignore = guest
if not self . source_mode and self . supports_property ( " source_mode " ) :
self . source_mode = self . MODE_BIND
2015-05-06 20:54:00 +03:00
2014-01-19 22:56:06 +04: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 23:27:27 +03:00
if not host :
host = " 127.0.0.1 "
2014-01-19 22:56:06 +04: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 17:14:37 +04:00
2013-07-24 22:37:07 +04:00
_XML_PROP_ORDER = [ " type " , " _has_mode_bind " , " _has_mode_connect " ,
2013-07-16 17:14:37 +04:00
" bind_host " , " bind_port " ,
2015-09-05 23:27:27 +03:00
" source_mode " , " source_host " , " source_port " ,
" _source_path " , " source_channel " ,
2013-07-16 17:14:37 +04:00
" target_type " , " target_name " ]
2018-02-23 04:44:09 +03:00
type = XMLProperty ( " ./@type " )
2013-09-24 17:25:05 +04:00
_tty = XMLProperty ( " ./@tty " )
2018-02-23 04:44:09 +03:00
_source_path = XMLProperty ( " ./source/@path " )
2013-09-24 17: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 17:14:37 +04:00
2018-02-23 04:44:09 +03:00
source_channel = XMLProperty ( " ./source/@channel " )
2016-08-24 23:37:36 +03:00
source_master = XMLProperty ( " ./source/@master " )
source_slave = XMLProperty ( " ./source/@slave " )
2014-03-25 18:42:32 +04:00
2015-05-06 20:54:00 +03:00
2015-09-05 23:27:27 +03:00
###################
# source handling #
###################
2015-05-06 20:54:00 +03:00
2015-09-05 23:27:27 +03:00
source_mode = XMLProperty ( " ./source/@mode " )
2013-07-16 17:14:37 +04:00
2013-09-19 21:27:30 +04:00
_has_mode_connect = XMLProperty ( " ./source[@mode= ' connect ' ]/@mode " )
_has_mode_bind = XMLProperty ( " ./source[@mode= ' bind ' ]/@mode " )
2013-07-16 17:14:37 +04:00
2015-09-05 23:27:27 +03:00
def _set_source_validate ( self , val ) :
if val is None :
return None
self . _has_mode_connect = self . MODE_CONNECT
return val
source_host = XMLProperty ( " ./source[@mode= ' connect ' ]/@host " ,
set_converter = _set_source_validate )
source_port = XMLProperty ( " ./source[@mode= ' connect ' ]/@service " ,
set_converter = _set_source_validate ,
is_int = True )
2013-07-16 17:14:37 +04:00
def _set_bind_validate ( self , val ) :
if val is None :
return None
2015-09-05 23:27:27 +03:00
self . _has_mode_bind = self . MODE_BIND
2013-07-16 17:14:37 +04:00
return val
2013-09-19 21:27:30 +04:00
bind_host = XMLProperty ( " ./source[@mode= ' bind ' ]/@host " ,
2013-07-16 17:14:37 +04:00
set_converter = _set_bind_validate )
2013-09-19 21:27:30 +04:00
bind_port = XMLProperty ( " ./source[@mode= ' bind ' ]/@service " ,
2015-09-05 23:27:27 +03:00
set_converter = _set_bind_validate ,
is_int = True )
#######################
# Remaining XML props #
#######################
2013-07-16 17:14:37 +04:00
def _get_default_protocol ( self ) :
if not self . supports_property ( " protocol " ) :
return None
return self . PROTOCOL_RAW
2013-09-19 21:27:30 +04:00
protocol = XMLProperty ( " ./protocol/@type " ,
2013-07-16 17:14:37 +04:00
default_cb = _get_default_protocol )
def _get_default_target_type ( self ) :
2018-03-21 01:59:14 +03:00
if self . DEVICE_TYPE == " channel " :
2013-07-16 17:14:37 +04:00
return self . CHANNEL_TARGET_VIRTIO
return None
2013-09-19 21:27:30 +04:00
target_type = XMLProperty ( " ./target/@type " ,
2013-07-16 17:14:37 +04:00
default_cb = _get_default_target_type )
2018-02-23 04:44:09 +03:00
target_address = XMLProperty ( " ./target/@address " )
2013-07-16 17:14:37 +04:00
2018-02-23 04:44:09 +03:00
target_port = XMLProperty ( " ./target/@port " , is_int = True )
2013-07-16 17:14:37 +04:00
def _default_target_name ( self ) :
if self . type == self . TYPE_SPICEVMC :
2013-10-05 21:27:11 +04:00
return self . CHANNEL_NAME_SPICE
2013-07-16 17:14:37 +04:00
return None
2013-09-19 21:27:30 +04:00
target_name = XMLProperty ( " ./target/@name " ,
2013-07-16 17:14:37 +04:00
default_cb = _default_target_name )
2013-04-13 22:34:52 +04:00
2016-07-18 22:03:06 +03:00
log_file = XMLProperty ( " ./log/@file " )
log_append = XMLProperty ( " ./log/@append " , is_onoff = True )
2013-04-13 22:34:52 +04:00
2018-03-20 19:18:35 +03:00
class DeviceConsole ( _DeviceChar ) :
2018-03-21 17:53:34 +03:00
XML_NAME = " console "
2018-03-20 19:18:35 +03:00
TYPES = [ _DeviceChar . TYPE_PTY ]
2013-04-13 22:34:52 +04:00
2018-03-20 19:18:35 +03:00
class DeviceSerial ( _DeviceChar ) :
2018-03-21 17:53:34 +03:00
XML_NAME = " serial "
2013-03-18 01:06:52 +04:00
2013-04-13 22:34:52 +04:00
2018-03-20 19:18:35 +03:00
class DeviceParallel ( _DeviceChar ) :
2018-03-21 17:53:34 +03:00
XML_NAME = " parallel "
2013-04-13 22:34:52 +04:00
2018-03-20 19:18:35 +03:00
class DeviceChannel ( _DeviceChar ) :
2018-03-21 17:53:34 +03:00
XML_NAME = " channel "
2018-03-20 19:18:35 +03:00
TYPES = ( _DeviceChar . _TYPES_FOR_CHANNEL +
_DeviceChar . _TYPES_FOR_ALL )