-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathrpc.py
127 lines (105 loc) · 4.34 KB
/
rpc.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
#!/usr/bin/env python
# -*- coding: utf-8 -*-
#
# rpc.py - Generic RPC Somewhat higher-level GUI_RPC API for BOINC core client
#
# Copyright (C) 2013 Rodrigo Silva (MestreLion) <[email protected]>
# Copyright (C) 2020 Jonathan Drake (drakej) <[email protected]>
#
# 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 3 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. See <http://www.gnu.org/licenses/gpl.html>
# A replacement of gui_rpc_client for basic RPC calls, with a sane API
import socket
from xml.etree import ElementTree
import logging
LOGGER = logging.getLogger('boinc-cluster')
GUI_RPC_HOSTNAME = None # localhost
GUI_RPC_PORT = 31416
GUI_RPC_TIMEOUT = 5
class Rpc(object):
''' Class to perform GUI RPC calls to a BOINC core client.
Usage in a context manager ('with' block) is recommended to ensure
disconnect() is called. Using the same instance for all calls is also
recommended so it reuses the same socket connection
'''
def __init__(self, hostname="", port=0, timeout=0, text_output=False):
self.hostname = hostname
self.port = port
self.timeout = timeout
self.sock = None
self.text_output = text_output
@property
def sockargs(self):
return (self.hostname, self.port, self.timeout)
def __enter__(self): self.connect(*self.sockargs); return self
def __exit__(self, *args): self.disconnect()
def connect(self, hostname="", port=0, timeout=0):
''' Connect to (hostname, port) with timeout in seconds.
Hostname defaults to None (localhost), and port to 31416
Calling multiple times will disconnect previous connection (if any),
and (re-)connect to host.
'''
if self.sock:
self.disconnect()
self.hostname = hostname or GUI_RPC_HOSTNAME
self.port = port or GUI_RPC_PORT
self.timeout = timeout or GUI_RPC_TIMEOUT
self.sock = socket.create_connection(
self.sockargs[0:2], self.sockargs[2])
def disconnect(self):
''' Disconnect from host. Calling multiple times is OK (idempotent)
'''
if self.sock:
self.sock.close()
self.sock = None
def call(self, request, text_output=None):
''' Do an RPC call. Pack and send the XML request and return the
unpacked reply. request can be either plain XML text or a
xml.etree.ElementTree.Element object. Return ElementTree.Element
or XML text according to text_output flag.
Will auto-connect if not connected.
'''
if text_output is None:
text_output = self.text_output
if not self.sock:
self.connect(*self.sockargs)
if not isinstance(request, ElementTree.Element):
request = ElementTree.fromstring(request)
# pack request
end = b'\003'
req = b"<boinc_gui_rpc_request>\n%s\n</boinc_gui_rpc_request>\n%s" \
% (ElementTree.tostring(request).replace(b' />', b'/>'), end)
try:
self.sock.sendall(req)
except (socket.error, socket.herror, socket.gaierror, socket.timeout):
raise
req = b""
while True:
try:
buf = self.sock.recv(8192)
if not buf:
raise socket.error("No data from socket")
except socket.error:
raise
n = buf.find(end)
if not n == -1:
break
req += buf
req += buf[:n]
# unpack reply (remove root tag, ie: first and last lines)
req = b"\n".join(req.strip().rsplit(b"\n")[1:-1])
LOGGER.debug(f"RPC {request.tag} call made on host {self.hostname}")
if text_output:
return req
else:
return ElementTree.fromstring(req)