-
Notifications
You must be signed in to change notification settings - Fork 0
/
flickrfs-ng.py
executable file
·145 lines (127 loc) · 4.97 KB
/
flickrfs-ng.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
#!/usr/bin/env python
from __future__ import with_statement
import logging, logging.handlers
import os
from errno import EACCES, ENOENT
from sys import argv, exit
from threading import Lock
from libs.fusepy.fuse import FUSE, FuseOSError, Operations, LoggingMixIn
from libs.python_flickr_api import flickr_api as flickr
from i_node import INode, MODE_DIR, MODE_FILE
from oauth_http_server import OAuthHTTPServer
from photo import PhotoStream, PhotoCache
from ConfigParser import ConfigParser
class Flickrfs(LoggingMixIn, Operations):
def __init__(self):
self.rwlock = Lock()
self.nodes = {}
self.__init_config()
self.__init_logging()
self.__init_fs()
self.__auth()
self.photo_stream = PhotoStream(self.nodes['/stream'], '/stream', self.user)
self.fd = 0
#self._get_photos()
def __init_logging(self):
logger = logging.getLogger()
handler = logging.handlers.RotatingFileHandler(self.log_file, "a", 5242880, 3)
formatter = logging.Formatter(
"%(asctime)s %(name)-14s %(levelname)-7s %(threadName)-10s %(funcName)-22s %(message)s", "%x %X")
handler.setFormatter(formatter)
logger.addHandler(handler)
logger.setLevel(logging.INFO)
def __init_fs(self):
self.nodes['/'] = INode.root()
self.nodes['/tags'] = self.nodes['/'].mknod(st_mode = MODE_DIR)
self.nodes['/tags/personal'] = self.nodes['/tags'].mknod(st_mode = MODE_DIR)
self.nodes['/tags/public'] = self.nodes['/tags'].mknod(st_mode = MODE_DIR)
self.nodes['/sets'] = self.nodes['/'].mknod(st_mode = MODE_DIR)
self.nodes['/date'] = self.nodes['/'].mknod(st_mode = MODE_DIR)
self.nodes['/stream'] = self.nodes['/'].mknod(st_mode = MODE_DIR)
self.nodes['/user'] = self.nodes['/'].mknod(st_mode = MODE_FILE)
def __init_config(self):
self.home = os.getenv('HOME')
self.config_dir = os.path.join(self.home, '.flickrfs-ng')
self.config_file = os.path.join(self.config_dir, 'config.txt')
self.auth_file = os.path.join(self.config_dir, 'auth.txt')
self.cache_file = os.path.join(self.config_dir, 'cache')
self.log_file = os.path.join(self.config_dir, 'flickrfs-ng.log')
PhotoCache.set_cache_file(self.cache_file)
self.browser = "/usr/bin/x-www-browser"
if os.path.exists(self.config_file):
config = ConfigParser()
config.read(self.config_file)
self.browser = config.get('flickrfs-ng', 'browser')
def __auth(self):
print "Start authenticate..."
if os.path.exists(self.auth_file):
a = flickr.auth.AuthHandler.load(self.auth_file)
else:
server = OAuthHTTPServer()
a = flickr.auth.AuthHandler(callback='http://localhost:%d/verifier' %
server.port)
pid = os.fork()
if pid == 0:
import webbrowser
webbrowser.open(a.get_authorization_url('write'))
#os.system("%s '%s'" % (self.browser, a.get_authorization_url('write')))
exit(0)
try:
server.serve_forever()
except IOError:
# safely ignore
pass
a.set_verifier(server.oauth_verifier)
a.save(self.auth_file)
flickr.set_auth_handler(a)
print "Login..."
self.user = flickr.test.login()
print "Authentication done."
def _get_photos(self):
photos = self.user.getPhotos(per_page=500)
print photos.info.pages
print photos.info.page
print photos.info.total
for photo in photos:
print photo.id, photo.title.encode('utf-8')
def getattr(self, path, fh=None):
if path.startswith('/stream/'):
attr = self.photo_stream.getattr(path, fh)
if not attr:
raise FuseOSError(ENOENT)
return attr
if path not in self.nodes.keys():
raise FuseOSError(ENOENT)
return self.nodes[path].getattrs()
def readdir(self, path, fh):
if path.startswith('/stream'):
return ['.', '..'] + self.photo_stream.file_list()
p = [x.rsplit('/', 1) for x in self.nodes.keys()]
if path == '/':
path = ''
return [t[1] for t in p if t[0] == path and len(t[1]) > 0] + ['.', '..']
def read(self, path, size, offset, fh):
log.info("%s offset %s length %s", path, offset, size)
if path.startswith('/stream'):
(parent, base) = path.rsplit('/', 1)
assert parent == self.photo_stream.path
return self.photo_stream.read(base, size, offset)
def open(self, path, flags):
# if open is called with read flag, it should notify the photo_stream to
# prefetch file content
log.info("path: %s flags: %d" % (path, flags))
if path.startswith('/stream'):
(parent, base) = path.rsplit('/', 1)
assert parent == self.photo_stream.path
if base not in self.photo_stream.file_list():
raise FuseOSError(ENOENT)
self.photo_stream.prefetch_file(base)
log.info("file size: %d" % self.photo_stream.photos[base].inode['st_size'])
self.fd += 1
return self.fd
if __name__ == '__main__':
if len(argv) != 2:
print('usage: %s <mountpoint>' % argv[0])
exit(1)
log = logging.getLogger()
fuse = FUSE(Flickrfs(), argv[1], foreground=True)