2014-04-07 16:43:39 -04:00
|
|
|
# vim: tabstop=4 shiftwidth=4 softtabstop=4
|
|
|
|
|
|
|
|
# Copyright 2014 IBM Corporation
|
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
2013-09-14 11:08:48 -04:00
|
|
|
# ALl rights reserved
|
|
|
|
|
|
|
|
# This is the socket api layer.
|
|
|
|
# It implement unix and tls sockets
|
2013-10-10 14:17:08 -04:00
|
|
|
#
|
2014-04-18 17:13:50 -04:00
|
|
|
|
2014-11-18 17:02:54 -05:00
|
|
|
import atexit
|
2014-04-18 17:13:50 -04:00
|
|
|
import os
|
|
|
|
import pwd
|
|
|
|
import stat
|
|
|
|
import struct
|
2014-10-28 13:42:21 -04:00
|
|
|
import sys
|
2014-04-18 17:13:50 -04:00
|
|
|
import traceback
|
|
|
|
|
|
|
|
import eventlet.green.socket as socket
|
|
|
|
import eventlet.green.ssl as ssl
|
|
|
|
import eventlet
|
|
|
|
|
2014-02-09 10:43:26 -05:00
|
|
|
import confluent.auth as auth
|
2014-05-06 13:37:31 -04:00
|
|
|
import confluent.tlvdata as tlvdata
|
2013-11-02 13:25:56 -04:00
|
|
|
import confluent.consoleserver as consoleserver
|
2013-11-02 16:58:38 -04:00
|
|
|
import confluent.config.configmanager as configmanager
|
2014-02-09 17:35:49 -05:00
|
|
|
import confluent.exceptions as exc
|
2014-04-18 10:36:51 -04:00
|
|
|
import confluent.log as log
|
2014-05-06 10:48:29 -04:00
|
|
|
import confluent.core as pluginapi
|
2014-04-18 17:13:50 -04:00
|
|
|
|
2013-10-14 09:21:55 -04:00
|
|
|
|
2014-04-18 10:36:51 -04:00
|
|
|
tracelog = None
|
|
|
|
auditlog = None
|
2015-01-13 15:01:55 -05:00
|
|
|
try:
|
|
|
|
SO_PEERCRED = socket.SO_PEERCRED
|
|
|
|
except AttributeError:
|
2015-02-10 17:16:10 -05:00
|
|
|
import platform
|
|
|
|
if "ppc64" in platform.machine():
|
|
|
|
SO_PEERCRED = 21
|
2015-02-27 16:26:27 -05:00
|
|
|
else:
|
2015-02-10 17:16:10 -05:00
|
|
|
SO_PEERCRED = 17
|
2013-09-14 11:08:48 -04:00
|
|
|
|
2014-04-18 17:13:50 -04:00
|
|
|
|
2014-02-09 10:43:26 -05:00
|
|
|
class ClientConsole(object):
|
|
|
|
def __init__(self, client):
|
2014-02-10 09:16:29 -05:00
|
|
|
self.client = client
|
|
|
|
self.xmit = False
|
2014-04-02 15:48:31 -04:00
|
|
|
self.pendingdata = []
|
2014-02-09 10:43:26 -05:00
|
|
|
|
|
|
|
def sendall(self, data):
|
2014-02-10 09:16:29 -05:00
|
|
|
if not self.xmit:
|
2014-04-02 15:48:31 -04:00
|
|
|
self.pendingdata.append(data)
|
2014-02-10 09:16:29 -05:00
|
|
|
return
|
2014-03-04 17:12:19 -05:00
|
|
|
tlvdata.send(self.client, data)
|
2014-02-09 10:43:26 -05:00
|
|
|
|
2014-02-10 09:16:29 -05:00
|
|
|
def startsending(self):
|
|
|
|
self.xmit = True
|
2014-04-02 15:48:31 -04:00
|
|
|
for datum in self.pendingdata:
|
|
|
|
tlvdata.send(self.client, datum)
|
2014-02-10 09:16:29 -05:00
|
|
|
self.pendingdata = None
|
|
|
|
|
|
|
|
|
2014-04-23 15:12:26 -04:00
|
|
|
def sessionhdl(connection, authname, skipauth=False):
|
2014-02-09 10:43:26 -05:00
|
|
|
# For now, trying to test the console stuff, so let's just do n4.
|
|
|
|
authenticated = False
|
2014-02-10 09:19:22 -05:00
|
|
|
authdata = None
|
2014-04-21 10:17:13 -04:00
|
|
|
cfm = None
|
2014-03-10 13:15:31 -04:00
|
|
|
if skipauth:
|
2014-02-09 10:43:26 -05:00
|
|
|
authenticated = True
|
|
|
|
cfm = configmanager.ConfigManager(tenant=None)
|
|
|
|
elif authname:
|
|
|
|
authdata = auth.authorize(authname, element=None)
|
2014-02-10 09:41:08 -05:00
|
|
|
if authdata is not None:
|
|
|
|
cfm = authdata[1]
|
|
|
|
authenticated = True
|
2014-04-18 17:13:50 -04:00
|
|
|
tlvdata.send(connection, "Confluent -- v0 --")
|
2014-02-09 10:43:26 -05:00
|
|
|
while not authenticated: # prompt for name and passphrase
|
2014-03-04 17:12:19 -05:00
|
|
|
tlvdata.send(connection, {'authpassed': 0})
|
|
|
|
response = tlvdata.recv(connection)
|
2014-03-09 20:34:39 -04:00
|
|
|
authname = response['username']
|
2014-07-27 19:23:32 -04:00
|
|
|
passphrase = response['password']
|
2014-02-09 17:35:49 -05:00
|
|
|
# note(jbjohnso): here, we need to authenticate, but not
|
2014-02-09 10:43:26 -05:00
|
|
|
# authorize a user. When authorization starts understanding
|
|
|
|
# element path, that authorization will need to be called
|
|
|
|
# per request the user makes
|
2014-03-09 20:34:39 -04:00
|
|
|
authdata = auth.check_user_passphrase(authname, passphrase)
|
2014-04-18 10:36:51 -04:00
|
|
|
if authdata is None:
|
|
|
|
auditlog.log(
|
|
|
|
{'operation': 'connect', 'user': authname, 'allowed': False})
|
|
|
|
else:
|
2014-02-09 10:43:26 -05:00
|
|
|
authenticated = True
|
|
|
|
cfm = authdata[1]
|
2014-03-04 17:12:19 -05:00
|
|
|
tlvdata.send(connection, {'authpassed': 1})
|
|
|
|
request = tlvdata.recv(connection)
|
2014-02-09 17:35:49 -05:00
|
|
|
while request is not None:
|
2014-03-03 15:52:12 -05:00
|
|
|
try:
|
2014-04-18 10:36:51 -04:00
|
|
|
process_request(
|
|
|
|
connection, request, cfm, authdata, authname, skipauth)
|
2014-04-21 10:17:13 -04:00
|
|
|
except exc.ForbiddenRequest:
|
2014-04-18 10:36:51 -04:00
|
|
|
tlvdata.send(connection, {'errorcode': 403,
|
2014-04-21 10:17:13 -04:00
|
|
|
'error': 'Forbidden'})
|
2014-04-18 10:36:51 -04:00
|
|
|
tlvdata.send(connection, {'_requestdone': 1})
|
2014-05-05 19:53:57 -04:00
|
|
|
except exc.TargetEndpointBadCredentials:
|
|
|
|
tlvdata.send(connection, {'errorcode': 502,
|
|
|
|
'error': 'Bad Credentials'})
|
|
|
|
tlvdata.send(connection, {'_requestdone': 1})
|
2015-03-26 09:24:23 -04:00
|
|
|
except exc.TargetEndpointUnreachable as tu:
|
2014-05-05 19:53:57 -04:00
|
|
|
tlvdata.send(connection, {'errorcode': 504,
|
2015-03-26 09:24:23 -04:00
|
|
|
'error': 'Unreachable Target - ' + str(
|
|
|
|
tu)})
|
2014-05-05 19:53:57 -04:00
|
|
|
tlvdata.send(connection, {'_requestdone': 1})
|
2014-11-25 13:57:31 -05:00
|
|
|
except exc.NotImplementedException:
|
|
|
|
tlvdata.send(connection, {'errorcode': 501,
|
|
|
|
'error': 'Not Implemented'})
|
|
|
|
tlvdata.send(connection, {'_requestdone': 1})
|
2015-03-23 13:48:39 -04:00
|
|
|
except exc.NotFoundException as nfe:
|
|
|
|
tlvdata.send(connection, {'errorcode': 404,
|
|
|
|
'error': str(nfe)})
|
|
|
|
tlvdata.send(connection, {'_requestdone': 1})
|
|
|
|
except exc.InvalidArgumentException as iae:
|
|
|
|
tlvdata.send(connection, {'errorcode': 400,
|
|
|
|
'error': 'Bad Request - ' + str(iae)})
|
|
|
|
tlvdata.send(connection, {'_requestdone': 1})
|
2014-10-28 13:42:21 -04:00
|
|
|
except SystemExit:
|
|
|
|
sys.exit(0)
|
2014-03-03 15:52:12 -05:00
|
|
|
except:
|
2014-04-18 10:36:51 -04:00
|
|
|
tracelog.log(traceback.format_exc(), ltype=log.DataTypes.event,
|
2014-04-21 10:17:13 -04:00
|
|
|
event=log.Events.stacktrace)
|
2014-03-04 17:12:19 -05:00
|
|
|
tlvdata.send(connection, {'errorcode': 500,
|
2014-04-21 10:17:13 -04:00
|
|
|
'error': 'Unexpected error'})
|
2014-03-04 17:12:19 -05:00
|
|
|
tlvdata.send(connection, {'_requestdone': 1})
|
|
|
|
request = tlvdata.recv(connection)
|
2014-02-09 17:35:49 -05:00
|
|
|
|
|
|
|
|
|
|
|
def send_response(responses, connection):
|
2014-02-09 19:26:48 -05:00
|
|
|
if responses is None:
|
|
|
|
return
|
2014-02-09 17:35:49 -05:00
|
|
|
for rsp in responses:
|
2014-03-04 17:12:19 -05:00
|
|
|
tlvdata.send(connection, rsp.raw())
|
|
|
|
tlvdata.send(connection, {'_requestdone': 1})
|
2014-02-09 19:26:48 -05:00
|
|
|
|
2014-02-09 17:35:49 -05:00
|
|
|
|
2014-04-18 10:36:51 -04:00
|
|
|
def process_request(connection, request, cfm, authdata, authname, skipauth):
|
|
|
|
if not isinstance(request, dict):
|
|
|
|
raise ValueError
|
|
|
|
operation = request['operation']
|
|
|
|
path = request['path']
|
|
|
|
params = request.get('parameters', None)
|
|
|
|
hdlr = None
|
|
|
|
if not skipauth:
|
|
|
|
authdata = auth.authorize(authdata[2], path, authdata[3], operation)
|
|
|
|
auditmsg = {
|
|
|
|
'operation': operation,
|
|
|
|
'user': authdata[2],
|
|
|
|
'target': path,
|
|
|
|
}
|
|
|
|
if authdata[3] is not None:
|
|
|
|
auditmsg['tenant'] = authdata[3]
|
|
|
|
if authdata is None:
|
|
|
|
auditmsg['allowed'] = False
|
|
|
|
auditlog.log(auditmsg)
|
|
|
|
raise exc.ForbiddenRequest()
|
|
|
|
auditmsg['allowed'] = True
|
|
|
|
auditlog.log(auditmsg)
|
|
|
|
try:
|
|
|
|
if operation == 'start':
|
|
|
|
elems = path.split('/')
|
|
|
|
if elems[3] != "console":
|
|
|
|
raise exc.InvalidArgumentException()
|
|
|
|
node = elems[2]
|
|
|
|
ccons = ClientConsole(connection)
|
2015-02-02 17:15:14 -05:00
|
|
|
skipreplay = False
|
|
|
|
if params and 'skipreplay' in params and params['skipreplay']:
|
|
|
|
skipreplay = True
|
2014-04-18 10:36:51 -04:00
|
|
|
consession = consoleserver.ConsoleSession(
|
|
|
|
node=node, configmanager=cfm, username=authname,
|
2015-02-02 17:15:14 -05:00
|
|
|
datacallback=ccons.sendall, skipreplay=skipreplay)
|
2014-04-18 10:36:51 -04:00
|
|
|
if consession is None:
|
|
|
|
raise Exception("TODO")
|
|
|
|
tlvdata.send(connection, {'started': 1})
|
|
|
|
ccons.startsending()
|
2014-10-06 15:12:16 -04:00
|
|
|
bufferage = consession.get_buffer_age()
|
|
|
|
if bufferage is not False:
|
|
|
|
tlvdata.send(connection, {'bufferage': bufferage})
|
2014-04-18 10:36:51 -04:00
|
|
|
while consession is not None:
|
|
|
|
data = tlvdata.recv(connection)
|
|
|
|
if type(data) == dict:
|
|
|
|
if data['operation'] == 'stop':
|
2014-02-10 09:16:29 -05:00
|
|
|
consession.destroy()
|
|
|
|
return
|
2014-04-18 10:36:51 -04:00
|
|
|
elif data['operation'] == 'break':
|
|
|
|
consession.send_break()
|
|
|
|
continue
|
2014-08-28 13:58:31 -04:00
|
|
|
elif data['operation'] == 'reopen':
|
|
|
|
consession.reopen()
|
|
|
|
continue
|
2014-04-18 10:36:51 -04:00
|
|
|
else:
|
|
|
|
raise Exception("TODO")
|
|
|
|
if not data:
|
|
|
|
consession.destroy()
|
|
|
|
return
|
|
|
|
consession.write(data)
|
2014-10-07 11:14:22 -04:00
|
|
|
elif operation == 'shutdown':
|
|
|
|
configmanager.ConfigManager.shutdown()
|
2014-04-18 10:36:51 -04:00
|
|
|
else:
|
|
|
|
hdlr = pluginapi.handle_path(path, operation, cfm, params)
|
2015-03-25 09:57:25 -04:00
|
|
|
except exc.NotFoundException as e:
|
2014-04-18 10:36:51 -04:00
|
|
|
tlvdata.send(connection, {"errorcode": 404,
|
2015-03-25 09:57:25 -04:00
|
|
|
"error": "Target not found - " + str(e)})
|
2014-04-18 10:36:51 -04:00
|
|
|
tlvdata.send(connection, {"_requestdone": 1})
|
2014-05-09 16:38:55 -04:00
|
|
|
except exc.InvalidArgumentException as e:
|
2014-04-18 10:36:51 -04:00
|
|
|
tlvdata.send(connection, {"errorcode": 400,
|
2014-05-09 16:38:55 -04:00
|
|
|
"error": "Bad Request - " + str(e)})
|
2014-04-18 10:36:51 -04:00
|
|
|
tlvdata.send(connection, {"_requestdone": 1})
|
|
|
|
send_response(hdlr, connection)
|
2014-02-09 17:35:49 -05:00
|
|
|
return
|
2013-09-14 11:08:48 -04:00
|
|
|
|
|
|
|
|
2013-10-14 09:21:55 -04:00
|
|
|
def _tlshandler():
|
2014-05-19 16:52:01 -04:00
|
|
|
plainsocket = socket.socket(socket.AF_INET6)
|
2013-10-09 20:14:34 -04:00
|
|
|
plainsocket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
|
2014-02-22 14:12:39 -05:00
|
|
|
plainsocket.setsockopt(socket.IPPROTO_TCP, socket.TCP_NODELAY, 1)
|
2014-05-20 09:41:55 -04:00
|
|
|
plainsocket.bind(('::', 13001, 0, 0))
|
|
|
|
plainsocket.listen(5)
|
|
|
|
while (1): # TODO: exithook
|
|
|
|
cnn, addr = plainsocket.accept()
|
|
|
|
eventlet.spawn_n(_tlsstartup, cnn)
|
|
|
|
|
|
|
|
|
|
|
|
def _tlsstartup(cnn):
|
|
|
|
authname = None
|
|
|
|
cnn = ssl.wrap_socket(cnn, keyfile="/etc/confluent/privkey.pem",
|
2014-04-21 10:17:13 -04:00
|
|
|
certfile="/etc/confluent/srvcert.pem",
|
|
|
|
ssl_version=ssl.PROTOCOL_TLSv1,
|
|
|
|
server_side=True)
|
2014-05-20 09:41:55 -04:00
|
|
|
sessionhdl(cnn, authname)
|
2013-09-14 11:08:48 -04:00
|
|
|
|
2014-11-18 17:02:54 -05:00
|
|
|
def removesocket():
|
|
|
|
try:
|
|
|
|
os.remove("/var/run/confluent/api.sock")
|
|
|
|
except OSError:
|
|
|
|
pass
|
2013-10-14 09:21:55 -04:00
|
|
|
|
|
|
|
def _unixdomainhandler():
|
|
|
|
unixsocket = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM)
|
|
|
|
try:
|
|
|
|
os.remove("/var/run/confluent/api.sock")
|
|
|
|
except OSError: # if file does not exist, no big deal
|
|
|
|
pass
|
|
|
|
unixsocket.bind("/var/run/confluent/api.sock")
|
2014-02-10 09:41:08 -05:00
|
|
|
os.chmod("/var/run/confluent/api.sock",
|
|
|
|
stat.S_IWOTH | stat.S_IROTH | stat.S_IWGRP |
|
|
|
|
stat.S_IRGRP | stat.S_IWUSR | stat.S_IRUSR)
|
2014-11-18 17:02:54 -05:00
|
|
|
atexit.register(removesocket)
|
2013-10-14 09:21:55 -04:00
|
|
|
unixsocket.listen(5)
|
2014-04-18 17:13:50 -04:00
|
|
|
while True:
|
2013-10-14 09:21:55 -04:00
|
|
|
cnn, addr = unixsocket.accept()
|
|
|
|
creds = cnn.getsockopt(socket.SOL_SOCKET, SO_PEERCRED,
|
2015-01-13 11:55:33 -05:00
|
|
|
struct.calcsize('iII'))
|
|
|
|
pid, uid, gid = struct.unpack('iII', creds)
|
2014-03-10 13:15:31 -04:00
|
|
|
skipauth = False
|
2013-10-14 11:28:07 -04:00
|
|
|
if uid in (os.getuid(), 0):
|
|
|
|
#this is where we happily accept the person
|
|
|
|
#to do whatever. This allows the server to
|
|
|
|
#start with no configuration whatsoever
|
|
|
|
#and yet still be configurable by some means
|
2014-03-10 13:15:31 -04:00
|
|
|
skipauth = True
|
|
|
|
try:
|
|
|
|
authname = pwd.getpwuid(uid).pw_name
|
|
|
|
except:
|
|
|
|
authname = "UNKNOWN SUPERUSER"
|
2013-10-14 11:28:07 -04:00
|
|
|
else:
|
|
|
|
try:
|
|
|
|
authname = pwd.getpwuid(uid).pw_name
|
|
|
|
except KeyError:
|
|
|
|
cnn.close()
|
|
|
|
return
|
2014-03-10 13:15:31 -04:00
|
|
|
eventlet.spawn_n(sessionhdl, cnn, authname, skipauth)
|
2013-10-14 09:21:55 -04:00
|
|
|
|
|
|
|
|
2013-09-14 11:08:48 -04:00
|
|
|
class SockApi(object):
|
2014-04-18 17:13:50 -04:00
|
|
|
def __init__(self):
|
|
|
|
self.tlsserver = None
|
|
|
|
self.unixdomainserver = None
|
|
|
|
|
2013-09-14 11:08:48 -04:00
|
|
|
def start(self):
|
2014-04-18 10:36:51 -04:00
|
|
|
global auditlog
|
|
|
|
global tracelog
|
|
|
|
tracelog = log.Logger('trace')
|
|
|
|
auditlog = log.Logger('audit')
|
2013-10-14 09:21:55 -04:00
|
|
|
self.tlsserver = eventlet.spawn(_tlshandler)
|
|
|
|
self.unixdomainserver = eventlet.spawn(_unixdomainhandler)
|