177 lines
6.9 KiB
Python
177 lines
6.9 KiB
Python
|
|
###########################################################################
|
||
|
|
#
|
||
|
|
# This program is part of Zenoss Core, an open source monitoring platform.
|
||
|
|
# Copyright (C) 2008-2010, Zenoss Inc.
|
||
|
|
#
|
||
|
|
# This program is free software; you can redistribute it and/or modify it
|
||
|
|
# under the terms of the GNU General Public License version 2, or (at your
|
||
|
|
# option) any later version, as published by the Free Software Foundation.
|
||
|
|
#
|
||
|
|
# For complete information please visit: http://www.zenoss.com/oss/
|
||
|
|
#
|
||
|
|
###########################################################################
|
||
|
|
|
||
|
|
__doc__="""
|
||
|
|
Define a simple API for performing a sequence of RPC calls to a
|
||
|
|
single host in an asynchronous fashion
|
||
|
|
"""
|
||
|
|
|
||
|
|
from pysamba.twisted.reactor import eventContext
|
||
|
|
from pysamba.library import library, logFuncCall
|
||
|
|
from pysamba.twisted.callback import Callback
|
||
|
|
from pysamba.composite_context import *
|
||
|
|
from pysamba.talloc import *
|
||
|
|
from pysamba.rpc.rpc_request import rpc_request
|
||
|
|
|
||
|
|
class dcerpc_pipe(Structure): pass
|
||
|
|
class dcerpc_interface_table(Structure): pass
|
||
|
|
|
||
|
|
# struct composite_context* dcerpc_pipe_connect_send(TALLOC_CTX *parent_ctx,
|
||
|
|
# const char *binding,
|
||
|
|
# const struct dcerpc_interface_table *table,
|
||
|
|
# struct cli_credentials *credentials,
|
||
|
|
# struct event_context *ev);
|
||
|
|
library.dcerpc_pipe_connect_send.restype = POINTER(composite_context)
|
||
|
|
library.dcerpc_pipe_connect_send.argtypes = [c_void_p, c_char_p, c_void_p, c_void_p, c_void_p]
|
||
|
|
library.dcerpc_pipe_connect_send = logFuncCall(library.dcerpc_pipe_connect_send)
|
||
|
|
|
||
|
|
# NTSTATUS dcerpc_pipe_connect_recv(struct composite_context *c,
|
||
|
|
# TALLOC_CTX *mem_ctx,
|
||
|
|
# struct dcerpc_pipe **pp);
|
||
|
|
library.dcerpc_pipe_connect_recv.restype = NTSTATUS
|
||
|
|
library.dcerpc_pipe_connect_recv.argtypes = [POINTER(composite_context), c_void_p, c_void_p]
|
||
|
|
library.dcerpc_pipe_connect_recv = logFuncCall(library.dcerpc_pipe_connect_recv)
|
||
|
|
|
||
|
|
# _PUBLIC_ NTSTATUS dcerpc_ndr_request_recv(struct rpc_request *req);
|
||
|
|
library.dcerpc_ndr_request_recv.restype = NTSTATUS
|
||
|
|
library.dcerpc_ndr_request_recv.argtypes = [c_void_p]
|
||
|
|
library.dcerpc_ndr_request_recv = logFuncCall(library.dcerpc_ndr_request_recv)
|
||
|
|
|
||
|
|
#_PUBLIC_ void composite_continue_rpc(struct composite_context *ctx,
|
||
|
|
# struct rpc_request *new_req,
|
||
|
|
# void (*continuation)(struct rpc_request *),
|
||
|
|
# void *private_data)
|
||
|
|
library.composite_continue_rpc.restype = None
|
||
|
|
library.composite_continue_rpc.argtypes = [POINTER(composite_context),
|
||
|
|
POINTER(rpc_request), c_void_p, c_void_p]
|
||
|
|
library.composite_continue_rpc = logFuncCall(library.composite_continue_rpc)
|
||
|
|
|
||
|
|
#
|
||
|
|
# Continue the RPC connect after a successful socket open to the server by
|
||
|
|
# receiving the results.
|
||
|
|
#
|
||
|
|
@logFuncCall
|
||
|
|
def rpc_connect_continue(ctx):
|
||
|
|
c = talloc_get_type(ctx.contents.async.private_data, composite_context)
|
||
|
|
if not library.composite_is_ok(c):
|
||
|
|
return
|
||
|
|
|
||
|
|
# complete the pipe connect and receive a pointer to the
|
||
|
|
# dcerpc_pipe structure
|
||
|
|
pipe = POINTER(dcerpc_pipe)()
|
||
|
|
import time
|
||
|
|
c.contents.status = library.dcerpc_pipe_connect_recv(ctx, c, byref(pipe));
|
||
|
|
c.contents.async.private_data = cast(pipe, c_void_p)
|
||
|
|
if not library.composite_is_ok(c):
|
||
|
|
return
|
||
|
|
library.composite_done(c)
|
||
|
|
continue_callback = CFUNCTYPE(None, POINTER(composite_context))
|
||
|
|
rpc_connect_continue = continue_callback(rpc_connect_continue)
|
||
|
|
|
||
|
|
#
|
||
|
|
# Open an RPC connection to the specified server
|
||
|
|
#
|
||
|
|
@logFuncCall
|
||
|
|
def async_rpc_open(event_ctx, server, cred, binding, arg, callback):
|
||
|
|
# create a composite context for this sequence of asynchronous calls
|
||
|
|
c = composite_create(None, event_ctx)
|
||
|
|
c.contents.async.fn = callback
|
||
|
|
c.contents.async.private_data = None
|
||
|
|
|
||
|
|
# build a binding string using the only protocol we care about...
|
||
|
|
binding = library.talloc_strdup(c, "%s:%s" % (binding, server))
|
||
|
|
|
||
|
|
# create a set of credentials
|
||
|
|
creds = library.cli_credentials_init(c)
|
||
|
|
if library.composite_nomem(creds, c):
|
||
|
|
return c
|
||
|
|
library.cli_credentials_set_conf(creds);
|
||
|
|
if cred:
|
||
|
|
library.cli_credentials_parse_string(creds, cred, CRED_SPECIFIED);
|
||
|
|
|
||
|
|
# issue the asynchronous rpc pipe connect request
|
||
|
|
rpc_ctx = library.dcerpc_pipe_connect_send(c,
|
||
|
|
binding,
|
||
|
|
arg,
|
||
|
|
creds,
|
||
|
|
event_ctx)
|
||
|
|
if library.composite_nomem(rpc_ctx, c):
|
||
|
|
return c
|
||
|
|
|
||
|
|
# setup the next stage of the connect process
|
||
|
|
library.composite_continue(c, rpc_ctx, rpc_connect_continue, c)
|
||
|
|
return c
|
||
|
|
|
||
|
|
# Fetch the return result after an RPC call completes
|
||
|
|
@logFuncCall
|
||
|
|
def continue_rpc(rpc_ctx):
|
||
|
|
c = talloc_get_type(rpc_ctx.contents.async.private, composite_context)
|
||
|
|
c.contents.async.private_data = rpc_ctx.contents.ndr.struct_ptr
|
||
|
|
c.contents.status = library.dcerpc_ndr_request_recv(rpc_ctx)
|
||
|
|
if not library.composite_is_ok(c):
|
||
|
|
return
|
||
|
|
library.composite_done(c)
|
||
|
|
continue_rpc_callback = CFUNCTYPE(None, POINTER(rpc_request))
|
||
|
|
continue_rpc = continue_rpc_callback(continue_rpc)
|
||
|
|
|
||
|
|
class Rpc(object):
|
||
|
|
|
||
|
|
def __init__(self):
|
||
|
|
self.ctx = self.rpc_pipe = None
|
||
|
|
|
||
|
|
def __del__(self):
|
||
|
|
self.close()
|
||
|
|
|
||
|
|
def close(self):
|
||
|
|
if self.ctx:
|
||
|
|
talloc_free(self.ctx)
|
||
|
|
self.ctx = self.rpc_pipe = None
|
||
|
|
|
||
|
|
def connect(self, host, credentials, object, binding='ncacn_np'):
|
||
|
|
table = dcerpc_interface_table.in_dll(library, 'dcerpc_table_' + object)
|
||
|
|
cb = Callback()
|
||
|
|
ctx = async_rpc_open(eventContext,
|
||
|
|
host,
|
||
|
|
credentials,
|
||
|
|
binding,
|
||
|
|
byref(table),
|
||
|
|
cb.callback)
|
||
|
|
cb.deferred.addCallback(self._store_rpc_pipe, ctx)
|
||
|
|
cb.deferred.addErrback(self._errback_cleanup, ctx)
|
||
|
|
return cb.deferred
|
||
|
|
|
||
|
|
def _errback_cleanup(self, result, ctx):
|
||
|
|
talloc_free(ctx)
|
||
|
|
return result
|
||
|
|
|
||
|
|
def _store_rpc_pipe(self, rpc_pipe, ctx):
|
||
|
|
self.ctx = ctx
|
||
|
|
self.rpc_pipe = rpc_pipe
|
||
|
|
return rpc_pipe
|
||
|
|
|
||
|
|
# ctx is the memory context to use to make the call, if not the one that
|
||
|
|
# started the original open call
|
||
|
|
@logFuncCall
|
||
|
|
def call(self, send_func, arg, ctx = None):
|
||
|
|
cb = Callback()
|
||
|
|
if ctx is None:
|
||
|
|
ctx = self.ctx
|
||
|
|
ctx.contents.async.fn = cb.callback
|
||
|
|
ctx.contents.async.private_data = None
|
||
|
|
rpc_ctx = cast(send_func(self.rpc_pipe, ctx, arg), POINTER(rpc_request))
|
||
|
|
if library.composite_nomem(rpc_ctx, ctx):
|
||
|
|
return
|
||
|
|
library.composite_continue_rpc(ctx, rpc_ctx, continue_rpc, ctx)
|
||
|
|
return cb.deferred
|
||
|
|
|