-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathnfs_nlm_wrapper.py
152 lines (133 loc) · 6.6 KB
/
nfs_nlm_wrapper.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
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
import sys
import rpyc
from logbook import Logger, FileHandler
from rpyc.utils.server import ThreadedServer
from mountclient import TCPMountClient
from nfsclient import NFSClient, NfsStat3, CreateMode, UnexpectedNfsStatus
from nlmclient import NLMClient, NLM4_Stats
from packer_arguments import get_packer_arguments
logger = Logger("nfs_client")
class FileNotFound(Exception):
pass
class NFSClientWrapper(rpyc.Service):
FILE_SYNC = 2
def on_connect(self, *args):
logger.info(f"Remote connection accepted")
def _get_export_handle(self, host, export):
mount_client = TCPMountClient(host)
sf = mount_client.mount(export)
export_handle = sf[1]
if export_handle:
return export_handle
def exposed_write_to_file(self, host, export, file_name, write_buffer, offset=0):
nfs_client = NFSClient(host=host)
file_handle = self.exposed_lookup_file(host, export, file_name)
if not file_handle:
file_handle = self.exposed_create_file(host, export, file_name)
write_arguments = get_packer_arguments("WRITE",
file=file_handle,
offset=offset,
count=len(write_buffer),
stable=NFSClientWrapper.FILE_SYNC,
data=write_buffer)
nfs_client.write(write_arguments)
return file_handle
def exposed_lookup_file(self, host, export, file_name):
logger.debug(f"Lookup for file {file_name} on export {export}, host {host}")
nfs_client = NFSClient(host=host)
lookup_args = get_packer_arguments("LOOKUP", dir=self._get_export_handle(host, export), name=file_name)
status, fh = nfs_client.lookup(lookup_args['what'])
if status == NfsStat3.NFS3_OK:
logger.debug("file {} was found".format(file_name))
else:
logger.debug("file {} was not found".format(file_name))
return fh
def exposed_create_file(self, host, export, file_name):
nfs_client = NFSClient(host=host)
create_args = get_packer_arguments(action_name="CREATE",
dir=self._get_export_handle(host, export),
name=file_name,
create_mode=CreateMode.UNCHECKED.value)
status, fh = nfs_client.create(create_args)
if status == NfsStat3.NFS3_OK:
logger.debug("file {} was successfully created".format(file_name))
return fh
def read_dirs(self, host, export):
nfs_client = NFSClient(host=host)
list_dir = []
read_dir_arguments = get_packer_arguments("READDIR",
dir=self._get_export_handle(host, export))
eof = False
while not eof:
status, rest = nfs_client.read_dir(read_dir_arguments)
if status != NfsStat3.NFS3_OK:
raise UnexpectedNfsStatus(NfsStat3(status))
entries, eof = rest
for fileid, name, cookie in entries:
list_dir.append((fileid, name))
return list_dir
def list_dir(self, host, export):
nfs_client = NFSClient(host=host)
list_dir = []
read_dir_plus_arguments = get_packer_arguments("READDIRPLUS",
dir=self._get_export_handle(host, export))
eof = False
while not eof:
status, attributes, rest = nfs_client.read_dir_plus(read_dir_plus_arguments)
if status != NfsStat3.NFS3_OK:
raise UnexpectedNfsStatus(NfsStat3(status))
entries, eof = rest
for file_id, dir_or_file_name, dir_params, cookie, fh in entries:
list_dir.append((file_id, dir_or_file_name))
return list_dir
def exposed_lock(self, host, export, file_name, owner, client_name, **kwargs):
exclusive = kwargs.get("exclusive", True)
block = kwargs.get("block", False)
offset = kwargs.get("offset", 0)
l_len = kwargs.get("length", 0)
file_handle = kwargs.get("file_handle")
logger.debug(
f"Locking the file {file_name} on host {host}, owner={owner}, client={client_name},"
f" kwargs={kwargs}")
nlm_client = NLMClient(host)
file_handle = self._get_file_handle(host, export,
file_name) if not file_handle else file_handle
lock_arguments = get_packer_arguments("LOCK",
caller_name=client_name,
block=block,
exclusive=exclusive,
fh=file_handle,
owner=owner,
l_offset=offset,
l_len=l_len)
status = nlm_client.lock(lock_arguments)
return NLM4_Stats(status).name
def exposed_unlock(self, host, export, file_name, owner, client_name, **kwargs):
offset = kwargs.get("offset", 0)
length = kwargs.get("length", 0)
file_handle = kwargs.get("file_handle")
logger.debug(
f"Unlocking the file {file_name} on host {host}, owner={owner}, client={client_name},"
f" kwargs = {kwargs}")
file_handle = self._get_file_handle(host, export,
file_name) if not file_handle else file_handle
nlm_client = NLMClient(host)
unlock_arguments = get_packer_arguments("UNLOCK",
caller_name=client_name,
owner=owner,
fh=file_handle,
l_offset=offset,
l_len=length)
status = nlm_client.unlock(unlock_arguments)
return NLM4_Stats(status).name
def _get_file_handle(self, host, export, file_name):
file_handle = self.exposed_lookup_file(host, export, file_name)
if not file_handle:
raise FileNotFound(f"{file_name} cannot be found and file_handle was not specified")
return file_handle
if __name__ == "__main__":
FileHandler("nfs_client.log").push_application()
t = ThreadedServer(NFSClientWrapper, port=9999)
t.daemon = True
logger.notice("Starting server on port 9999")
t.start()