openlp/openlp/core/lib/projector/pjlink1.py

959 lines
39 KiB
Python
Raw Normal View History

2014-10-06 19:10:03 +00:00
# -*- coding: utf-8 -*-
# vim: autoindent shiftwidth=4 expandtab textwidth=120 tabstop=4 softtabstop=4
###############################################################################
# OpenLP - Open Source Lyrics Projection #
# --------------------------------------------------------------------------- #
2015-12-31 22:46:06 +00:00
# Copyright (c) 2008-2016 OpenLP Developers #
2014-10-06 19:10:03 +00:00
# --------------------------------------------------------------------------- #
# 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; version 2 of the License. #
# #
# 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., 59 #
# Temple Place, Suite 330, Boston, MA 02111-1307 USA #
###############################################################################
"""
2014-10-17 17:28:12 +00:00
:mod:`openlp.core.lib.projector.pjlink1` module
Provides the necessary functions for connecting to a PJLink-capable projector.
2014-10-06 19:10:03 +00:00
2014-10-17 17:28:12 +00:00
See PJLink Class 1 Specifications for details.
http://pjlink.jbmia.or.jp/english/dl.html
2015-09-08 19:13:59 +00:00
2014-10-17 17:28:12 +00:00
Section 5-1 PJLink Specifications
2015-09-08 19:13:59 +00:00
2014-10-17 17:28:12 +00:00
Section 5-5 Guidelines for Input Terminals
2014-10-06 19:10:03 +00:00
NOTE:
Function names follow the following syntax:
def process_CCCC(...):
WHERE:
CCCC = PJLink command being processed.
"""
import logging
log = logging.getLogger(__name__)
2014-10-21 21:40:47 +00:00
log.debug('pjlink1 loaded')
2014-10-06 19:10:03 +00:00
__all__ = ['PJLink1']
2014-10-17 17:28:12 +00:00
from codecs import decode
2014-10-06 19:10:03 +00:00
2015-11-07 00:49:40 +00:00
from PyQt5.QtCore import pyqtSignal, pyqtSlot
from PyQt5.QtNetwork import QAbstractSocket, QTcpSocket
2014-10-06 19:10:03 +00:00
2016-06-18 03:18:52 +00:00
from openlp.core.common import translate, md5_hash
2014-10-06 19:10:03 +00:00
from openlp.core.lib.projector.constants import *
# Shortcuts
SocketError = QAbstractSocket.SocketError
SocketSTate = QAbstractSocket.SocketState
PJLINK_PREFIX = '%'
PJLINK_CLASS = '1'
2016-06-18 03:18:52 +00:00
PJLINK_HEADER = '{prefix}{linkclass}'.format(prefix=PJLINK_PREFIX, linkclass=PJLINK_CLASS)
2014-10-06 19:10:03 +00:00
PJLINK_SUFFIX = CR
class PJLink1(QTcpSocket):
"""
Socket service for connecting to a PJLink-capable projector.
"""
2014-10-17 17:28:12 +00:00
# Signals sent by this module
2014-10-06 19:10:03 +00:00
changeStatus = pyqtSignal(str, int, str)
projectorNetwork = pyqtSignal(int) # Projector network activity
2014-10-15 17:22:12 +00:00
projectorStatus = pyqtSignal(int) # Status update
2014-10-13 20:16:25 +00:00
projectorAuthentication = pyqtSignal(str) # Authentication error
projectorNoAuthentication = pyqtSignal(str) # PIN set and no authentication needed
2014-10-15 17:22:12 +00:00
projectorReceivedData = pyqtSignal() # Notify when received data finished processing
projectorUpdateIcons = pyqtSignal() # Update the status icons on toolbar
2014-10-06 19:10:03 +00:00
def __init__(self, name=None, ip=None, port=PJLINK_PORT, pin=None, *args, **kwargs):
"""
Setup for instance.
:param name: Display name
:param ip: IP address to connect to
:param port: Port to use. Default to PJLINK_PORT
:param pin: Access pin (if needed)
Optional parameters
:param dbid: Database ID number
:param location: Location where projector is physically located
:param notes: Extra notes about the projector
2014-10-21 21:24:16 +00:00
:param poll_time: Time (in seconds) to poll connected projector
:param socket_timeout: Time (in seconds) to abort the connection if no response
2014-10-06 19:10:03 +00:00
"""
2016-06-18 03:18:52 +00:00
log.debug('PJlink(args={args} kwargs={kwargs})'.format(args=args, kwargs=kwargs))
2014-10-06 19:10:03 +00:00
self.name = name
self.ip = ip
self.port = port
self.pin = pin
super(PJLink1, self).__init__()
self.dbid = None
self.location = None
self.notes = None
2014-10-23 02:21:26 +00:00
self.dbid = None if 'dbid' not in kwargs else kwargs['dbid']
2016-01-03 08:15:54 +00:00
self.location = None if 'location' not in kwargs else kwargs['location']
2014-10-23 02:21:26 +00:00
self.notes = None if 'notes' not in kwargs else kwargs['notes']
# Poll time 20 seconds unless called with something else
self.poll_time = 20000 if 'poll_time' not in kwargs else kwargs['poll_time'] * 1000
# Timeout 5 seconds unless called with something else
self.socket_timeout = 5000 if 'socket_timeout' not in kwargs else kwargs['socket_timeout'] * 1000
2014-10-21 23:45:15 +00:00
# In case we're called from somewhere that only wants information
self.no_poll = 'no_poll' in kwargs
2014-10-06 19:10:03 +00:00
self.i_am_running = False
self.status_connect = S_NOT_CONNECTED
self.last_command = ''
self.projector_status = S_NOT_CONNECTED
self.error_status = S_OK
# Socket information
2014-10-21 21:24:16 +00:00
# Add enough space to input buffer for extraneous \n \r
2014-10-06 19:10:03 +00:00
self.maxSize = PJLINK_MAX_PACKET + 2
self.setReadBufferSize(self.maxSize)
2014-10-21 21:24:16 +00:00
# PJLink information
2014-10-06 19:10:03 +00:00
self.pjlink_class = '1' # Default class
self.reset_information()
2014-10-06 19:10:03 +00:00
# Set from ProjectorManager.add_projector()
self.widget = None # QListBox entry
self.timer = None # Timer that calls the poll_loop
2014-10-15 17:22:12 +00:00
self.send_queue = []
self.send_busy = False
2014-10-21 21:24:16 +00:00
# Socket timer for some possible brain-dead projectors or network cable pulled
self.socket_timer = None
2014-10-17 17:28:12 +00:00
# Map command to function
2014-10-06 19:10:03 +00:00
self.PJLINK1_FUNC = {'AVMT': self.process_avmt,
'CLSS': self.process_clss,
'ERST': self.process_erst,
'INFO': self.process_info,
'INF1': self.process_inf1,
'INF2': self.process_inf2,
'INPT': self.process_inpt,
'INST': self.process_inst,
'LAMP': self.process_lamp,
'NAME': self.process_name,
2014-10-13 16:40:58 +00:00
'PJLINK': self.check_login,
2014-10-06 19:10:03 +00:00
'POWR': self.process_powr
}
def reset_information(self):
2014-10-17 17:28:12 +00:00
"""
Reset projector-specific information to default
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) reset_information() connect status is {state}'.format(ip=self.ip, state=self.state()))
self.power = S_OFF
self.pjlink_name = None
self.manufacturer = None
self.model = None
self.shutter = None
self.mute = None
self.lamp = None
self.fan = None
self.source_available = None
self.source = None
self.other_info = None
2014-10-13 16:40:58 +00:00
if hasattr(self, 'timer'):
2016-06-18 03:18:52 +00:00
log.debug('({ip}): Calling timer.stop()'.format(ip=self.ip))
2014-10-13 16:40:58 +00:00
self.timer.stop()
2014-10-17 02:28:51 +00:00
if hasattr(self, 'socket_timer'):
2016-06-18 03:18:52 +00:00
log.debug('({ip}): Calling socket_timer.stop()'.format(ip=self.ip))
2014-10-17 02:28:51 +00:00
self.socket_timer.stop()
self.send_queue = []
self.send_busy = False
2014-10-06 19:10:03 +00:00
def thread_started(self):
"""
Connects signals to methods when thread is started.
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Thread starting'.format(ip=self.ip))
2014-10-06 19:10:03 +00:00
self.i_am_running = True
self.connected.connect(self.check_login)
self.disconnected.connect(self.disconnect_from_host)
self.error.connect(self.get_error)
def thread_stopped(self):
"""
Cleanups when thread is stopped.
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Thread stopped'.format(ip=self.ip))
try:
self.connected.disconnect(self.check_login)
except TypeError:
pass
try:
self.disconnected.disconnect(self.disconnect_from_host)
except TypeError:
pass
try:
self.error.disconnect(self.get_error)
except TypeError:
pass
try:
self.projectorReceivedData.disconnect(self._send_command)
except TypeError:
pass
2014-10-06 19:10:03 +00:00
self.disconnect_from_host()
self.deleteLater()
self.i_am_running = False
2014-10-17 02:28:51 +00:00
def socket_abort(self):
"""
Aborts connection and closes socket in case of brain-dead projectors.
2014-10-17 17:28:12 +00:00
Should normally be called by socket_timer().
2014-10-17 02:28:51 +00:00
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) socket_abort() - Killing connection'.format(ip=self.ip))
2014-10-17 02:28:51 +00:00
self.disconnect_from_host(abort=True)
2014-10-06 19:10:03 +00:00
def poll_loop(self):
"""
2014-10-17 17:28:12 +00:00
Retrieve information from projector that changes.
Normally called by timer().
2014-10-06 19:10:03 +00:00
"""
if self.state() != self.ConnectedState:
return
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Updating projector status'.format(ip=self.ip))
2014-10-06 19:10:03 +00:00
# Reset timer in case we were called from a set command
if self.timer.interval() < self.poll_time:
# Reset timer to 5 seconds
self.timer.setInterval(self.poll_time)
2014-10-21 21:24:16 +00:00
# Restart timer
2014-10-06 19:10:03 +00:00
self.timer.start()
2014-10-21 21:24:16 +00:00
# These commands may change during connetion
2014-10-09 01:43:34 +00:00
for command in ['POWR', 'ERST', 'LAMP', 'AVMT', 'INPT']:
self.send_command(command, queue=True)
2014-10-21 21:24:16 +00:00
# The following commands do not change, so only check them once
2014-10-09 01:43:34 +00:00
if self.power == S_ON and self.source_available is None:
self.send_command('INST', queue=True)
if self.other_info is None:
self.send_command('INFO', queue=True)
2014-10-13 16:40:58 +00:00
if self.manufacturer is None:
self.send_command('INF1', queue=True)
2014-10-13 20:16:25 +00:00
if self.model is None:
self.send_command('INF2', queue=True)
2014-10-13 20:16:25 +00:00
if self.pjlink_name is None:
self.send_command('NAME', queue=True)
2014-10-13 20:16:25 +00:00
if self.power == S_ON and self.source_available is None:
self.send_command('INST', queue=True)
2014-10-06 19:10:03 +00:00
def _get_status(self, status):
"""
Helper to retrieve status/error codes and convert to strings.
2014-10-17 17:28:12 +00:00
:param status: Status/Error code
:returns: (Status/Error code, String)
2014-10-06 19:10:03 +00:00
"""
if status in ERROR_STRING:
2014-10-21 21:40:47 +00:00
return ERROR_STRING[status], ERROR_MSG[status]
2014-10-06 19:10:03 +00:00
elif status in STATUS_STRING:
2014-10-21 21:40:47 +00:00
return STATUS_STRING[status], ERROR_MSG[status]
2014-10-06 19:10:03 +00:00
else:
2014-10-21 21:40:47 +00:00
return status, translate('OpenLP.PJLink1', 'Unknown status')
2014-10-06 19:10:03 +00:00
def change_status(self, status, msg=None):
"""
Check connection/error status, set status for projector, then emit status change signal
for gui to allow changing the icons.
2014-10-17 17:28:12 +00:00
:param status: Status code
:param msg: Optional message
2014-10-06 19:10:03 +00:00
"""
2014-10-09 01:43:34 +00:00
message = translate('OpenLP.PJLink1', 'No message') if msg is None else msg
2014-10-06 19:10:03 +00:00
(code, message) = self._get_status(status)
if msg is not None:
message = msg
if status in CONNECTION_ERRORS:
# Projector, connection state
self.projector_status = self.error_status = self.status_connect = E_NOT_CONNECTED
elif status >= S_NOT_CONNECTED and status < S_STATUS:
self.status_connect = status
self.projector_status = S_NOT_CONNECTED
elif status < S_NETWORK_SENDING:
self.status_connect = S_CONNECTED
self.projector_status = status
(status_code, status_message) = self._get_status(self.status_connect)
2016-06-18 03:18:52 +00:00
log.debug('({ip}) status_connect: {code}: "{message}"'.format(ip=self.ip,
code=status_code,
message=status_message if msg is None else msg))
2014-10-06 19:10:03 +00:00
(status_code, status_message) = self._get_status(self.projector_status)
2016-06-18 03:18:52 +00:00
log.debug('({ip}) projector_status: {code}: "{message}"'.format(ip=self.ip,
code=status_code,
message=status_message if msg is None else msg))
2014-10-06 19:10:03 +00:00
(status_code, status_message) = self._get_status(self.error_status)
2016-06-18 03:18:52 +00:00
log.debug('({ip}) error_status: {code}: "{message}"'.format(ip=self.ip,
code=status_code,
message=status_message if msg is None else msg))
2014-10-06 19:10:03 +00:00
self.changeStatus.emit(self.ip, status, message)
2014-10-09 20:30:07 +00:00
@pyqtSlot()
def check_login(self, data=None):
2014-10-06 19:10:03 +00:00
"""
Processes the initial connection and authentication (if needed).
2014-10-17 17:28:12 +00:00
Starts poll timer if connection is established.
2016-06-18 03:18:52 +00:00
NOTE: Qt md5 hash function doesn't work with projector authentication. Use the python md5 hash function.
2014-10-17 17:28:12 +00:00
:param data: Optional data if called from another routine
2014-10-06 19:10:03 +00:00
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) check_login(data="{data}")'.format(ip=self.ip, data=data))
2014-10-09 20:30:07 +00:00
if data is None:
# Reconnected setup?
if not self.waitForReadyRead(2000):
# Possible timeout issue
2016-06-18 03:18:52 +00:00
log.error('({ip}) Socket timeout waiting for login'.format(ip=self.ip))
self.change_status(E_SOCKET_TIMEOUT)
return
2014-10-09 20:30:07 +00:00
read = self.readLine(self.maxSize)
dontcare = self.readLine(self.maxSize) # Clean out the trailing \r\n
2014-10-13 20:16:25 +00:00
if read is None:
2016-06-18 03:18:52 +00:00
log.warn('({ip}) read is None - socket error?'.format(ip=self.ip))
2014-10-13 20:16:25 +00:00
return
elif len(read) < 8:
2016-06-18 03:18:52 +00:00
log.warn('({ip}) Not enough data read)'.format(ip=self.ip))
2014-10-09 20:30:07 +00:00
return
data = decode(read, 'ascii')
# Possibility of extraneous data on input when reading.
# Clean out extraneous characters in buffer.
dontcare = self.readLine(self.maxSize)
2016-06-18 03:18:52 +00:00
log.debug('({ip}) check_login() read "{data}"'.format(ip=self.ip, data=data.strip()))
2014-10-06 19:10:03 +00:00
# At this point, we should only have the initial login prompt with
# possible authentication
2014-10-21 21:24:16 +00:00
# PJLink initial login will be:
# 'PJLink 0' - Unauthenticated login - no extra steps required.
# 'PJLink 1 XXXXXX' Authenticated login - extra processing required.
2014-10-06 19:10:03 +00:00
if not data.upper().startswith('PJLINK'):
# Invalid response
return self.disconnect_from_host()
2014-10-13 16:40:58 +00:00
if '=' in data:
2014-10-21 21:24:16 +00:00
# Processing a login reply
2014-10-13 16:40:58 +00:00
data_check = data.strip().split('=')
else:
2014-10-21 21:24:16 +00:00
# Process initial connection
2014-10-13 16:40:58 +00:00
data_check = data.strip().split(' ')
2016-06-18 03:18:52 +00:00
log.debug('({ip}) data_check="{data}"'.format(ip=self.ip, data=data_check))
2014-10-21 21:24:16 +00:00
# Check for projector reporting an error
2014-10-13 20:16:25 +00:00
if data_check[1].upper() == 'ERRA':
# Authentication error
self.disconnect_from_host()
self.change_status(E_AUTHENTICATION)
2016-06-18 03:18:52 +00:00
log.debug('({ip}) emitting projectorAuthentication() signal'.format(ip=self.name))
2014-10-13 20:16:25 +00:00
return
elif data_check[1] == '0' and self.pin is not None:
# Pin set and no authentication needed
2016-06-18 03:18:52 +00:00
log.warning('({ip}) Regular connection but PIN set'.format(ip=self.name))
2014-10-13 20:16:25 +00:00
self.disconnect_from_host()
self.change_status(E_AUTHENTICATION)
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Emitting projectorNoAuthentication() signal'.format(ip=self.name))
2014-10-13 20:16:25 +00:00
self.projectorNoAuthentication.emit(self.name)
return
elif data_check[1] == '1':
2014-10-06 19:10:03 +00:00
# Authenticated login with salt
2016-06-18 03:18:52 +00:00
if self.pin is None:
log.warning('({ip}) Authenticated connection but no pin set'.format(ip=self.name))
self.disconnect_from_host()
self.change_status(E_AUTHENTICATION)
log.debug('({ip}) Emitting projectorAuthentication() signal'.format(ip=self.name))
self.projectorAuthentication.emit(self.name)
return
else:
log.debug('({ip}) Setting hash with salt="{data}"'.format(ip=self.ip, data=data_check[2]))
log.debug('({ip}) pin="{data}"'.format(ip=self.ip, data=self.pin))
salt = md5_hash(salt=data_check[2].encode('ascii'), data=self.pin.encode('ascii'))
2014-10-13 16:40:58 +00:00
else:
salt = None
2014-10-06 19:10:03 +00:00
# We're connected at this point, so go ahead and do regular I/O
self.readyRead.connect(self.get_data)
2014-10-21 21:24:16 +00:00
self.projectorReceivedData.connect(self._send_command)
2014-10-06 19:10:03 +00:00
# Initial data we should know about
self.send_command(cmd='CLSS', salt=salt)
self.waitForReadyRead()
2014-10-21 23:45:15 +00:00
if (not self.no_poll) and (self.state() == self.ConnectedState):
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Starting timer'.format(ip=self.ip))
self.timer.setInterval(2000) # Set 2 seconds for initial information
2014-10-06 19:10:03 +00:00
self.timer.start()
2014-10-15 18:18:00 +00:00
@pyqtSlot()
2014-10-06 19:10:03 +00:00
def get_data(self):
"""
Socket interface to retrieve data.
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) get_data(): Reading data'.format(ip=self.ip))
2014-10-06 19:10:03 +00:00
if self.state() != self.ConnectedState:
2016-06-18 03:18:52 +00:00
log.debug('({ip}) get_data(): Not connected - returning'.format(ip=self.ip))
2014-10-23 02:21:26 +00:00
self.send_busy = False
2014-10-06 19:10:03 +00:00
return
read = self.readLine(self.maxSize)
if read == -1:
# No data available
2016-06-18 03:18:52 +00:00
log.debug('({ip}) get_data(): No data available (-1)'.format(ip=self.ip))
2014-10-23 02:21:26 +00:00
self.send_busy = False
2014-10-15 17:22:12 +00:00
self.projectorReceivedData.emit()
2014-10-06 19:10:03 +00:00
return
2014-10-17 02:28:51 +00:00
self.socket_timer.stop()
2014-10-06 19:10:03 +00:00
self.projectorNetwork.emit(S_NETWORK_RECEIVED)
data_in = decode(read, 'ascii')
data = data_in.strip()
if len(data) < 7:
2014-10-06 19:10:03 +00:00
# Not enough data for a packet
2016-06-18 03:18:52 +00:00
log.debug('({ip}) get_data(): Packet length < 7: "{data}"'.format(ip=self.ip, data=data))
2014-10-23 02:21:26 +00:00
self.send_busy = False
2014-10-15 17:22:12 +00:00
self.projectorReceivedData.emit()
2014-10-06 19:10:03 +00:00
return
2016-06-18 03:18:52 +00:00
log.debug('({ip}) get_data(): Checking new data "{data}"'.format(ip=self.ip, data=data))
2014-10-09 20:30:07 +00:00
if data.upper().startswith('PJLINK'):
# Reconnected from remote host disconnect ?
2014-10-15 17:22:12 +00:00
self.check_login(data)
2014-10-23 02:21:26 +00:00
self.send_busy = False
2014-10-15 17:22:12 +00:00
self.projectorReceivedData.emit()
return
elif '=' not in data:
2016-06-18 03:18:52 +00:00
log.warn('({ip}) get_data(): Invalid packet received'.format(ip=self.ip))
2014-10-23 02:21:26 +00:00
self.send_busy = False
2014-10-15 17:22:12 +00:00
self.projectorReceivedData.emit()
2014-10-06 19:10:03 +00:00
return
data_split = data.split('=')
try:
(prefix, class_, cmd, data) = (data_split[0][0], data_split[0][1], data_split[0][2:], data_split[1])
except ValueError as e:
2016-06-18 03:18:52 +00:00
log.warn('({ip}) get_data(): Invalid packet - expected header + command + data'.format(ip=self.ip))
log.warn('({ip}) get_data(): Received data: "{data}"'.format(ip=self.ip, data=data_in.strip()))
2014-10-06 19:10:03 +00:00
self.change_status(E_INVALID_DATA)
2014-10-23 02:21:26 +00:00
self.send_busy = False
2014-10-15 17:22:12 +00:00
self.projectorReceivedData.emit()
2014-10-06 19:10:03 +00:00
return
2014-10-23 02:21:26 +00:00
if not (self.pjlink_class in PJLINK_VALID_CMD and cmd in PJLINK_VALID_CMD[self.pjlink_class]):
2016-06-18 03:18:52 +00:00
log.warn('({ip}) get_data(): Invalid packet - unknown command "{data}"'.format(ip=self.ip, data=cmd))
2014-10-23 02:21:26 +00:00
self.send_busy = False
2014-10-15 17:22:12 +00:00
self.projectorReceivedData.emit()
2014-10-06 19:10:03 +00:00
return
return self.process_command(cmd, data)
@pyqtSlot(int)
def get_error(self, err):
"""
2014-10-17 17:28:12 +00:00
Process error from SocketError signal.
Remaps system error codes to projector error codes.
:param err: Error code
2014-10-06 19:10:03 +00:00
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) get_error(err={error}): {data}'.format(ip=self.ip, error=err, data=self.errorString()))
2014-10-06 19:10:03 +00:00
if err <= 18:
2014-10-13 16:40:58 +00:00
# QSocket errors. Redefined in projector.constants so we don't mistake
2014-10-06 19:10:03 +00:00
# them for system errors
check = err + E_CONNECTION_REFUSED
self.timer.stop()
else:
check = err
if check < E_GENERAL:
# Some system error?
self.change_status(err, self.errorString())
else:
self.change_status(E_NETWORK, self.errorString())
2014-10-15 18:18:00 +00:00
self.projectorUpdateIcons.emit()
2014-10-17 02:28:51 +00:00
if self.status_connect == E_NOT_CONNECTED:
self.abort()
self.reset_information()
2014-10-06 19:10:03 +00:00
return
def send_command(self, cmd, opts='?', salt=None, queue=False):
2014-10-06 19:10:03 +00:00
"""
2014-10-17 17:28:12 +00:00
Add command to output queue if not already in queue.
:param cmd: Command to send
2014-10-21 21:24:16 +00:00
:param opts: Command option (if any) - defaults to '?' (get information)
:param salt: Optional salt for md5 hash initial authentication
2014-10-17 17:28:12 +00:00
:param queue: Option to force add to queue rather than sending directly
2014-10-06 19:10:03 +00:00
"""
if self.state() != self.ConnectedState:
2016-06-18 03:18:52 +00:00
log.warn('({ip}) send_command(): Not connected - returning'.format(ip=self.ip))
2014-10-15 17:22:12 +00:00
self.send_queue = []
2014-10-06 19:10:03 +00:00
return
self.projectorNetwork.emit(S_NETWORK_SENDING)
2016-06-18 03:18:52 +00:00
log.debug('({ip}) send_command(): Building cmd="{command}" opts="{data}"{salt}'.format(ip=self.ip,
command=cmd,
data=opts,
salt='' if salt is None
else ' with hash'))
out = '{salt}{header}{command} {options}{suffix}'.format(salt="" if salt is None else salt,
header=PJLINK_HEADER,
command=cmd,
options=opts,
suffix=CR)
2014-10-15 17:22:12 +00:00
if out in self.send_queue:
# Already there, so don't add
2016-06-18 03:18:52 +00:00
log.debug('({ip}) send_command(out="{data}") Already in queue - skipping'.format(ip=self.ip,
data=out.strip()))
2014-10-15 20:27:43 +00:00
elif not queue and len(self.send_queue) == 0:
2014-10-21 21:24:16 +00:00
# Nothing waiting to send, so just send it
2016-06-18 03:18:52 +00:00
log.debug('({ip}) send_command(out="{data}") Sending data'.format(ip=self.ip, data=out.strip()))
2014-10-17 02:28:51 +00:00
return self._send_command(data=out)
2014-10-15 17:22:12 +00:00
else:
2016-06-18 03:18:52 +00:00
log.debug('({ip}) send_command(out="{data}") adding to queue'.format(ip=self.ip, data=out.strip()))
2014-10-15 17:22:12 +00:00
self.send_queue.append(out)
2014-10-17 02:28:51 +00:00
self.projectorReceivedData.emit()
2016-06-18 03:18:52 +00:00
log.debug('({ip}) send_command(): send_busy is {data}'.format(ip=self.ip, data=self.send_busy))
2014-10-15 20:27:43 +00:00
if not self.send_busy:
2016-06-18 03:18:52 +00:00
log.debug('({ip}) send_command() calling _send_string()'.format(ip=self.ip))
2014-10-17 02:28:51 +00:00
self._send_command()
2014-10-15 17:22:12 +00:00
@pyqtSlot()
2014-10-17 02:28:51 +00:00
def _send_command(self, data=None):
2014-10-15 17:22:12 +00:00
"""
Socket interface to send data. If data=None, then check queue.
:param data: Immediate data to send
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) _send_string()'.format(ip=self.ip))
log.debug('({ip}) _send_string(): Connection status: {data}'.format(ip=self.ip, data=self.state()))
2014-10-15 18:18:00 +00:00
if self.state() != self.ConnectedState:
2016-06-18 03:18:52 +00:00
log.debug('({ip}) _send_string() Not connected - abort'.format(ip=self.ip))
2014-10-15 18:18:00 +00:00
self.send_queue = []
self.send_busy = False
return
2014-10-16 20:03:20 +00:00
if self.send_busy:
# Still waiting for response from last command sent
return
2014-10-15 17:22:12 +00:00
if data is not None:
out = data
2016-06-18 03:18:52 +00:00
log.debug('({ip}) _send_string(data="{data}")'.format(ip=self.ip, data=out.strip()))
2014-10-15 17:22:12 +00:00
elif len(self.send_queue) != 0:
out = self.send_queue.pop(0)
2016-06-18 03:18:52 +00:00
log.debug('({ip}) _send_string(queued data="{data}"%s)'.format(ip=self.ip, data=out.strip()))
2014-10-15 17:22:12 +00:00
else:
# No data to send
2016-06-18 03:18:52 +00:00
log.debug('({ip}) _send_string(): No data to send'.format(ip=self.ip))
2014-10-15 17:22:12 +00:00
self.send_busy = False
return
self.send_busy = True
2016-06-18 03:18:52 +00:00
log.debug('({ip}) _send_string(): Sending "{data}"'.format(ip=self.ip, data=out.strip()))
log.debug('({ip}) _send_string(): Queue = {data}'.format(ip=self.ip, data=self.send_queue))
2014-10-17 02:28:51 +00:00
self.socket_timer.start()
2016-06-18 03:18:52 +00:00
self.projectorNetwork.emit(S_NETWORK_SENDING)
sent = self.write(out.encode('ascii'))
self.waitForBytesWritten(2000) # 2 seconds should be enough
if sent == -1:
# Network error?
self.change_status(E_NETWORK,
translate('OpenLP.PJLink1', 'Error while sending data to projector'))
2014-10-06 19:10:03 +00:00
def process_command(self, cmd, data):
"""
Verifies any return error code. Calls the appropriate command handler.
2014-10-17 17:28:12 +00:00
:param cmd: Command to process
:param data: Data being processed
2014-10-06 19:10:03 +00:00
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Processing command "{data}"'.format(ip=self.ip, data=cmd))
2014-10-06 19:10:03 +00:00
if data in PJLINK_ERRORS:
# Oops - projector error
2016-06-18 03:18:52 +00:00
log.error('({ip}) Projector returned error "{data}"'.format(ip=self.ip, data=data))
2014-10-06 19:10:03 +00:00
if data.upper() == 'ERRA':
# Authentication error
2014-10-13 16:40:58 +00:00
self.disconnect_from_host()
2014-10-06 19:10:03 +00:00
self.change_status(E_AUTHENTICATION)
2016-06-18 03:18:52 +00:00
log.debug('({ip}) emitting projectorAuthentication() signal'.format(ip=self.ip))
2014-10-13 17:09:23 +00:00
self.projectorAuthentication.emit(self.name)
2014-10-06 19:10:03 +00:00
elif data.upper() == 'ERR1':
# Undefined command
2016-06-18 03:18:52 +00:00
self.change_status(E_UNDEFINED, '{error} "{data}"'.format(error=translate('OpenLP.PJLink1',
'Undefined command:'),
data=cmd))
2014-10-06 19:10:03 +00:00
elif data.upper() == 'ERR2':
# Invalid parameter
self.change_status(E_PARAMETER)
elif data.upper() == 'ERR3':
# Projector busy
self.change_status(E_UNAVAILABLE)
elif data.upper() == 'ERR4':
# Projector/display error
self.change_status(E_PROJECTOR)
2014-10-15 20:27:43 +00:00
self.send_busy = False
2014-10-16 20:03:20 +00:00
self.projectorReceivedData.emit()
2014-10-15 20:27:43 +00:00
return
2014-10-06 19:10:03 +00:00
# Command succeeded - no extra information
2014-10-15 20:27:43 +00:00
elif data.upper() == 'OK':
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Command returned OK'.format(ip=self.ip))
2014-10-15 02:57:13 +00:00
# A command returned successfully, recheck data
2014-10-15 20:27:43 +00:00
self.send_busy = False
2014-10-16 20:03:20 +00:00
self.projectorReceivedData.emit()
2014-10-15 20:27:43 +00:00
return
2014-10-06 19:10:03 +00:00
if cmd in self.PJLINK1_FUNC:
2014-10-15 17:22:12 +00:00
self.PJLINK1_FUNC[cmd](data)
2014-10-06 19:10:03 +00:00
else:
2016-06-18 03:18:52 +00:00
log.warn('({ip}) Invalid command {data}'.format(ip=self.ip, data=cmd))
2014-10-15 20:27:43 +00:00
self.send_busy = False
2014-10-15 17:22:12 +00:00
self.projectorReceivedData.emit()
2014-10-06 19:10:03 +00:00
def process_lamp(self, data):
"""
Lamp(s) status. See PJLink Specifications for format.
2014-10-17 17:28:12 +00:00
Data may have more than 1 lamp to process.
Update self.lamp dictionary with lamp status.
:param data: Lamp(s) status.
2014-10-06 19:10:03 +00:00
"""
lamps = []
data_dict = data.split()
while data_dict:
try:
fill = {'Hours': int(data_dict[0]), 'On': False if data_dict[1] == '0' else True}
except ValueError:
# In case of invalid entry
2016-06-18 03:18:52 +00:00
log.warn('({ip}) process_lamp(): Invalid data "{data}"'.format(ip=self.ip, data=data))
return
2014-10-06 19:10:03 +00:00
lamps.append(fill)
data_dict.pop(0) # Remove lamp hours
data_dict.pop(0) # Remove lamp on/off
self.lamp = lamps
return
def process_powr(self, data):
"""
Power status. See PJLink specification for format.
2014-10-17 17:28:12 +00:00
Update self.power with status. Update icons if change from previous setting.
:param data: Power status
2014-10-06 19:10:03 +00:00
"""
if data in PJLINK_POWR_STATUS:
2014-10-15 17:22:12 +00:00
power = PJLINK_POWR_STATUS[data]
update_icons = self.power != power
self.power = power
2014-10-06 19:10:03 +00:00
self.change_status(PJLINK_POWR_STATUS[data])
2014-10-15 17:22:12 +00:00
if update_icons:
self.projectorUpdateIcons.emit()
# Update the input sources available
if power == S_ON:
self.send_command('INST')
2014-10-06 19:10:03 +00:00
else:
# Log unknown status response
2016-06-18 03:18:52 +00:00
log.warn('({ip}) Unknown power response: {data}'.format(ip=self.ip, data=data))
2014-10-06 19:10:03 +00:00
return
def process_avmt(self, data):
"""
2014-10-17 17:28:12 +00:00
Process shutter and speaker status. See PJLink specification for format.
Update self.mute (audio) and self.shutter (video shutter).
:param data: Shutter and audio status
2014-10-06 19:10:03 +00:00
"""
2014-10-15 17:22:12 +00:00
shutter = self.shutter
mute = self.mute
2014-10-06 19:10:03 +00:00
if data == '11':
2014-10-15 17:22:12 +00:00
shutter = True
mute = False
2014-10-06 19:10:03 +00:00
elif data == '21':
2014-10-15 17:22:12 +00:00
shutter = False
mute = True
2014-10-06 19:10:03 +00:00
elif data == '30':
2014-10-15 17:22:12 +00:00
shutter = False
mute = False
2014-10-06 19:10:03 +00:00
elif data == '31':
2014-10-15 17:22:12 +00:00
shutter = True
mute = True
2014-10-06 19:10:03 +00:00
else:
2016-06-18 03:18:52 +00:00
log.warn('({ip}) Unknown shutter response: {data}'.format(ip=self.ip, data=data))
2014-10-15 17:22:12 +00:00
update_icons = shutter != self.shutter
update_icons = update_icons or mute != self.mute
self.shutter = shutter
self.mute = mute
if update_icons:
self.projectorUpdateIcons.emit()
2014-10-06 19:10:03 +00:00
return
def process_inpt(self, data):
"""
Current source input selected. See PJLink specification for format.
2014-10-17 17:28:12 +00:00
Update self.source
:param data: Currently selected source
2014-10-06 19:10:03 +00:00
"""
self.source = data
2016-06-18 03:18:52 +00:00
log.info('({ip}) Setting data source to "{data}"'.format(ip=self.ip, data=self.source))
2014-10-06 19:10:03 +00:00
return
def process_clss(self, data):
"""
PJLink class that this projector supports. See PJLink specification for format.
2014-10-17 17:28:12 +00:00
Updates self.class.
:param data: Class that projector supports.
2014-10-06 19:10:03 +00:00
"""
# bug 1550891: Projector returns non-standard class response:
# : Expected: %1CLSS=1
# : Received: %1CLSS=Class 1
if len(data) > 1:
# Split non-standard information from response
clss = data.split()[-1]
else:
clss = data
self.pjlink_class = clss
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Setting pjlink_class for this projector to "{data}"'.format(ip=self.ip,
data=self.pjlink_class))
2014-10-06 19:10:03 +00:00
return
def process_name(self, data):
"""
2014-10-17 17:28:12 +00:00
Projector name set in projector.
Updates self.pjlink_name
:param data: Projector name
2014-10-06 19:10:03 +00:00
"""
self.pjlink_name = data
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Setting projector PJLink name to "{data}"'.format(ip=self.ip, data=self.pjlink_name))
2014-10-06 19:10:03 +00:00
return
def process_inf1(self, data):
"""
2014-10-17 17:28:12 +00:00
Manufacturer name set in projector.
Updates self.manufacturer
:param data: Projector manufacturer
2014-10-06 19:10:03 +00:00
"""
self.manufacturer = data
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Setting projector manufacturer data to "{data}"'.format(ip=self.ip, data=self.manufacturer))
2014-10-06 19:10:03 +00:00
return
def process_inf2(self, data):
"""
2014-10-17 17:28:12 +00:00
Projector Model set in projector.
Updates self.model.
:param data: Model name
2014-10-06 19:10:03 +00:00
"""
self.model = data
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Setting projector model to "{data}"'.format(ip=self.ip, data=self.model))
2014-10-06 19:10:03 +00:00
return
def process_info(self, data):
"""
2014-10-17 17:28:12 +00:00
Any extra info set in projector.
Updates self.other_info.
:param data: Projector other info
2014-10-06 19:10:03 +00:00
"""
self.other_info = data
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Setting projector other_info to "{data}"'.format(ip=self.ip, data=self.other_info))
2014-10-06 19:10:03 +00:00
return
def process_inst(self, data):
"""
Available source inputs. See PJLink specification for format.
2014-10-17 17:28:12 +00:00
Updates self.source_available
:param data: Sources list
2014-10-06 19:10:03 +00:00
"""
sources = []
check = data.split()
for source in check:
sources.append(source)
sources.sort()
2014-10-06 19:10:03 +00:00
self.source_available = sources
self.projectorUpdateIcons.emit()
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Setting projector sources_available to "{data}"'.format(ip=self.ip,
data=self.source_available))
2014-10-06 19:10:03 +00:00
return
def process_erst(self, data):
"""
Error status. See PJLink Specifications for format.
2014-10-17 17:28:12 +00:00
Updates self.projector_errors
:param data: Error status
2014-10-06 19:10:03 +00:00
"""
2014-10-15 17:34:27 +00:00
try:
2014-10-15 17:38:39 +00:00
datacheck = int(data)
2014-10-15 17:34:27 +00:00
except ValueError:
# Bad data - ignore
return
2014-10-15 17:38:39 +00:00
if datacheck == 0:
2014-10-06 19:10:03 +00:00
self.projector_errors = None
else:
self.projector_errors = {}
# Fan
if data[0] != '0':
self.projector_errors[translate('OpenLP.ProjectorPJLink', 'Fan')] = \
PJLINK_ERST_STATUS[data[0]]
# Lamp
if data[1] != '0':
self.projector_errors[translate('OpenLP.ProjectorPJLink', 'Lamp')] = \
PJLINK_ERST_STATUS[data[1]]
# Temp
if data[2] != '0':
self.projector_errors[translate('OpenLP.ProjectorPJLink', 'Temperature')] = \
PJLINK_ERST_STATUS[data[2]]
# Cover
if data[3] != '0':
self.projector_errors[translate('OpenLP.ProjectorPJLink', 'Cover')] = \
PJLINK_ERST_STATUS[data[3]]
# Filter
if data[4] != '0':
self.projector_errors[translate('OpenLP.ProjectorPJLink', 'Filter')] = \
PJLINK_ERST_STATUS[data[4]]
# Other
if data[5] != '0':
self.projector_errors[translate('OpenLP.ProjectorPJLink', 'Other')] = \
PJLINK_ERST_STATUS[data[5]]
return
def connect_to_host(self):
"""
2014-10-17 17:28:12 +00:00
Initiate connection to projector.
2014-10-06 19:10:03 +00:00
"""
if self.state() == self.ConnectedState:
2016-06-18 03:18:52 +00:00
log.warn('({ip}) connect_to_host(): Already connected - returning'.format(ip=self.ip))
2014-10-06 19:10:03 +00:00
return
self.change_status(S_CONNECTING)
self.connectToHost(self.ip, self.port if type(self.port) is int else int(self.port))
@pyqtSlot()
2014-10-17 02:28:51 +00:00
def disconnect_from_host(self, abort=False):
2014-10-06 19:10:03 +00:00
"""
Close socket and cleanup.
"""
2014-10-17 02:28:51 +00:00
if abort or self.state() != self.ConnectedState:
if abort:
2016-06-18 03:18:52 +00:00
log.warn('({ip}) disconnect_from_host(): Aborting connection'.format(ip=self.ip))
2014-10-17 02:28:51 +00:00
else:
2016-06-18 03:18:52 +00:00
log.warn('({ip}) disconnect_from_host(): Not connected - returning'.format(ip=self.ip))
2014-10-17 02:28:51 +00:00
self.reset_information()
2014-10-06 19:10:03 +00:00
self.disconnectFromHost()
try:
self.readyRead.disconnect(self.get_data)
except TypeError:
pass
2014-10-17 02:28:51 +00:00
if abort:
self.change_status(E_NOT_CONNECTED)
else:
2016-06-18 03:18:52 +00:00
log.debug('({ip}) disconnect_from_host() '
'Current status {data}'.format(ip=self.ip, data=self._get_status(self.status_connect)[0]))
2014-10-17 03:09:42 +00:00
if self.status_connect != E_NOT_CONNECTED:
self.change_status(S_NOT_CONNECTED)
2014-10-13 16:40:58 +00:00
self.reset_information()
self.projectorUpdateIcons.emit()
2014-10-06 19:10:03 +00:00
def get_available_inputs(self):
"""
Send command to retrieve available source inputs.
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Sending INST command'.format(ip=self.ip))
2014-10-06 19:10:03 +00:00
return self.send_command(cmd='INST')
def get_error_status(self):
"""
Send command to retrieve currently known errors.
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Sending ERST command'.format(ip=self.ip))
2014-10-06 19:10:03 +00:00
return self.send_command(cmd='ERST')
def get_input_source(self):
"""
Send command to retrieve currently selected source input.
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Sending INPT command'.format(ip=self.ip))
2014-10-06 19:10:03 +00:00
return self.send_command(cmd='INPT')
def get_lamp_status(self):
"""
Send command to return the lap status.
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Sending LAMP command'.format(ip=self.ip))
2014-10-06 19:10:03 +00:00
return self.send_command(cmd='LAMP')
def get_manufacturer(self):
"""
Send command to retrieve manufacturer name.
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Sending INF1 command'.format(ip=self.ip))
2014-10-06 19:10:03 +00:00
return self.send_command(cmd='INF1')
def get_model(self):
"""
Send command to retrieve the model name.
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Sending INF2 command'.format(ip=self.ip))
2014-10-06 19:10:03 +00:00
return self.send_command(cmd='INF2')
def get_name(self):
"""
Send command to retrieve name as set by end-user (if set).
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Sending NAME command'.format(ip=self.ip))
2014-10-06 19:10:03 +00:00
return self.send_command(cmd='NAME')
def get_other_info(self):
"""
Send command to retrieve extra info set by manufacturer.
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Sending INFO command'.format(ip=self.ip))
2014-10-06 19:10:03 +00:00
return self.send_command(cmd='INFO')
def get_power_status(self):
"""
Send command to retrieve power status.
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Sending POWR command'.format(ip=self.ip))
2014-10-06 19:10:03 +00:00
return self.send_command(cmd='POWR')
def get_shutter_status(self):
"""
2014-10-09 20:30:07 +00:00
Send command to retrieve shutter status.
2014-10-06 19:10:03 +00:00
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Sending AVMT command'.format(ip=self.ip))
2014-10-06 19:10:03 +00:00
return self.send_command(cmd='AVMT')
def set_input_source(self, src=None):
"""
Verify input source available as listed in 'INST' command,
then send the command to select the input source.
2014-10-17 17:28:12 +00:00
:param src: Video source to select in projector
2014-10-06 19:10:03 +00:00
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) set_input_source(src="{data}")'.format(ip=self.ip, data=src))
2014-10-06 19:10:03 +00:00
if self.source_available is None:
return
elif src not in self.source_available:
return
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Setting input source to "{data}"'.format(ip=self.ip, data=src))
2014-10-06 19:10:03 +00:00
self.send_command(cmd='INPT', opts=src)
2014-10-15 17:29:15 +00:00
self.poll_loop()
2014-10-06 19:10:03 +00:00
def set_power_on(self):
"""
Send command to turn power to on.
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Setting POWR to 1 (on)'.format(ip=self.ip))
2014-10-06 19:10:03 +00:00
self.send_command(cmd='POWR', opts='1')
2014-10-15 17:29:15 +00:00
self.poll_loop()
2014-10-06 19:10:03 +00:00
def set_power_off(self):
"""
Send command to turn power to standby.
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Setting POWR to 0 (standby)'.format(ip=self.ip))
2014-10-06 19:10:03 +00:00
self.send_command(cmd='POWR', opts='0')
2014-10-15 17:29:15 +00:00
self.poll_loop()
2014-10-06 19:10:03 +00:00
def set_shutter_closed(self):
"""
Send command to set shutter to closed position.
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Setting AVMT to 11 (shutter closed)'.format(ip=self.ip))
2014-10-06 19:10:03 +00:00
self.send_command(cmd='AVMT', opts='11')
2014-10-15 17:29:15 +00:00
self.poll_loop()
2014-10-06 19:10:03 +00:00
def set_shutter_open(self):
"""
Send command to set shutter to open position.
"""
2016-06-18 03:18:52 +00:00
log.debug('({ip}) Setting AVMT to "10" (shutter open)'.format(ip=self.ip))
2014-10-06 19:10:03 +00:00
self.send_command(cmd='AVMT', opts='10')
2014-10-15 17:29:15 +00:00
self.poll_loop()