-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathdfs.py
executable file
·139 lines (120 loc) · 3.39 KB
/
dfs.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
#!/usr/bin/env python3
from errno import ENOENT
from stat import S_IFDIR
try:
from fuse import FUSE, FuseOSError, Operations
except ImportError:
import sys
print(
"fusepy is required to mount DFS as unix filesystem.\n"
"It seems like you don't have it installed.\n"
"Install it for current user "
"(Run `pip install --user fusepy`) [Y/n]?",
end='',
file=sys.stderr,
)
answer = input()
if answer not in ['', 'Y', 'y']:
print("Not installing", file=sys.stderr)
sys.exit()
import os
code = os.system('pip install --user fusepy')
if code != 0:
print("Failed to install", file=sys.stderr)
sys.exit(code)
os.execv(sys.argv[0], sys.argv)
from http_name_node import HttpNameNode
class DFS(Operations):
def __init__(self, url, mkfs=False):
self._node = HttpNameNode(url)
if not self._node.ping_alive():
raise Exception('Cannot connect to cluster')
if mkfs:
self._node.mkfs()
def create(self, path, mode):
try:
self._node.touch(path)
return 0
except:
raise FuseOSError(ENOENT)
def getattr(self, path, fh=None):
try:
st = self._node.stat(path)
return {
'st_size': st[1],
'st_mode': st[2],
'st_nlink': 1,
}
except:
raise FuseOSError(ENOENT)
getxattr = None
def truncate(self, path, length, fh=None):
pass
def mkdir(self, path, mode):
try:
self._node.mkdir(path)
except:
raise FuseOSError(ENOENT)
def read(self, path, size, offset, fh):
try:
return self._node.cat(path)
except:
raise FuseOSError(ENOENT)
def readdir(self, path, fh):
try:
return self._node.ls(path)
except:
raise FuseOSError(ENOENT)
def readlink(self, path):
try:
return self._node.stat(path)[0]
except:
raise FuseOSError(ENOENT)
def rename(self, old, new):
try:
self._node.mv(old, new)
except:
raise FuseOSError(ENOENT)
def rmdir(self, path):
try:
self._node.rmdir(path)
except:
raise FuseOSError(ENOENT)
def unlink(self, path):
try:
self._node.rm(path)
except:
raise FuseOSError(ENOENT)
def write(self, path, data, offset, fh):
try:
self._node.tee(path, data)
return len(data)
except:
raise FuseOSError(ENOENT)
if __name__ == '__main__':
import argparse
parser = argparse.ArgumentParser(
description='DFS mount tool'
)
parser.add_argument(
'--mkfs',
action='store_true',
help='Initialize filesystem, erase all existing data',
)
parser.add_argument('mount', help='Mount directory')
parser.add_argument('host', help='Name node hostname')
parser.add_argument(
'port',
nargs='?',
default='8180',
help='Port to use for connection (default: 8180)',
)
args = parser.parse_args()
url = f'http://{args.host}:{args.port}/'
fuse = FUSE(
DFS(url, mkfs=args.mkfs),
args.mount,
foreground=False,
nothreads=True,
allow_other=True,
)