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
|
# coding: utf-8
# nslcd.py - functions for doing nslcd requests
#
# Copyright (C) 2013 Arthur de Jong
#
# This library is free software; you can redistribute it and/or
# modify it under the terms of the GNU Lesser General Public
# License as published by the Free Software Foundation; either
# version 2.1 of the License, or (at your option) any later version.
#
# This library 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
# Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public
# License along with this library; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
# 02110-1301 USA
import os
import socket
import struct
import fcntl
import constants
# definition for reading and writing INT32 values
_int32 = struct.Struct('!i')
class NslcdClient(object):
def __init__(self, action):
# set up the socket (store in class to avoid closing it)
self.sock = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM)
#fcntl.fcntl(sock, fcntl.F_SETFD, fcntl.FD_CLOEXEC)
# connect to nslcd
self.sock.connect(constants.NSLCD_SOCKET)
#self.sock.setblocking(1)
self.fp = os.fdopen(self.sock.fileno(), 'r+b', 1024 * 1024)
# write a request header with a request code
self.action = action
self.write_int32(constants.NSLCD_VERSION)
self.write_int32(action)
def write(self, value):
self.fp.write(value)
def write_int32(self, value):
self.write(_int32.pack(value))
def write_string(self, value):
self.write_int32(len(value))
self.write(value)
def write_ether(self, value):
value = struct.pack('BBBBBB', *(int(x, 16) for x in value.split(':')))
self.write(value)
def write_address(self, af, value):
self.write_int32(af)
self.write_string(value)
def read(self, size):
return self.fp.read(size)
def read_int32(self):
return _int32.unpack(self.read(_int32.size))[0]
def read_string(self):
len = self.read_int32()
return self.read(len)
def read_stringlist(self):
len = self.read_int32()
return [self.read_string() for x in xrange(len)]
def read_ether(self):
value = self.fp.read(6)
return ':'.join('%x' % x for x in struct.unpack('6B', value))
def read_address(self):
af = self.read_int32()
return af, socket.inet_ntop(af, self.read_string())
def read_addresslist(self):
len = self.read_int32()
return [self.read_address() for x in xrange(len)]
def get_response(self):
# complete the request if required and check response header
if self.action:
# flush the stream
self.fp.flush()
# read and check response version number
assert self.read_int32() == constants.NSLCD_VERSION
assert self.read_int32() == self.action
self.action = None
# get the NSLCD_RESULT_* marker and return it
return self.read_int32()
def close(self):
if hasattr(self, 'fp'):
try:
self.fp.close()
except IOError:
pass
def __del__(self):
self.close()
|