|
1
|
|
|
"""Functions to interact with sharing services.""" |
|
2
|
|
|
|
|
3
|
1 |
|
import os |
|
4
|
1 |
|
import re |
|
5
|
1 |
|
import logging |
|
6
|
|
|
|
|
7
|
|
|
|
|
8
|
1 |
|
ROOTS = ( |
|
9
|
|
|
r"C:\Users", |
|
10
|
|
|
r"/Users", |
|
11
|
|
|
r"/home", |
|
12
|
|
|
) |
|
13
|
1 |
|
SERVICES = ( |
|
14
|
|
|
'Dropbox', |
|
15
|
|
|
'Dropbox (Personal)', |
|
16
|
|
|
) |
|
17
|
1 |
|
CONFIG = 'mine.yml' |
|
18
|
1 |
|
CONFLICT_BASE = r"{} \(.+'s conflicted copy \d+-\d+-\d+.*\).*" |
|
19
|
1 |
|
CONFLICT_ANY = CONFLICT_BASE.format(".+") |
|
20
|
1 |
|
CONFLICT_CONFIG = CONFLICT_BASE.format("mine") |
|
21
|
1 |
|
DEPTH = 3 # number of levels to search for the settings file |
|
22
|
|
|
|
|
23
|
1 |
|
log = logging.getLogger(__name__) |
|
24
|
|
|
|
|
25
|
|
|
|
|
26
|
1 |
|
def find_root(top=None): |
|
27
|
|
|
"""Get the root of the shared directory.""" |
|
28
|
1 |
|
top = top or os.path.expanduser('~') |
|
29
|
|
|
|
|
30
|
1 |
|
log.debug("Looking for sharing service in '%s'...", top) |
|
31
|
1 |
|
for directory in os.listdir(top): |
|
32
|
1 |
|
if directory in SERVICES: |
|
33
|
1 |
|
path = os.path.join(top, directory) |
|
34
|
1 |
|
log.debug("Found sharing service: %s", path) |
|
35
|
1 |
|
return path |
|
36
|
|
|
|
|
37
|
1 |
|
msg = "no sharing service found" |
|
38
|
1 |
|
if os.getenv('CI'): |
|
39
|
1 |
|
log.warning(msg) |
|
40
|
1 |
|
return top |
|
41
|
|
|
else: |
|
42
|
1 |
|
raise EnvironmentError(msg) |
|
43
|
|
|
|
|
44
|
|
|
|
|
45
|
1 |
|
def find_config_path(top=None, root=None): |
|
46
|
|
|
"""Get the path to the settings file.""" |
|
47
|
1 |
|
log.info("Looking for settings file...") |
|
48
|
1 |
|
top = top or os.path.expanduser('~') |
|
49
|
1 |
|
root = root or find_root(top=top) |
|
50
|
|
|
|
|
51
|
1 |
|
log.debug("Looking for '%s' in '%s'...", CONFIG, root) |
|
52
|
1 |
|
for dirpath, dirnames, _, in os.walk(root): |
|
53
|
1 |
|
depth = dirpath.count(os.path.sep) - root.count(os.path.sep) |
|
54
|
1 |
|
if depth >= DEPTH: |
|
55
|
1 |
|
del dirnames[:] |
|
56
|
1 |
|
continue |
|
57
|
1 |
|
path = os.path.join(dirpath, CONFIG) |
|
58
|
1 |
|
if os.path.isfile(path) and \ |
|
59
|
|
|
not os.path.isfile(os.path.join(path, 'setup.py')): |
|
60
|
1 |
|
log.info("Found settings file: %s", path) |
|
61
|
1 |
|
return path |
|
62
|
|
|
|
|
63
|
1 |
|
raise EnvironmentError("No '{}' file found".format(CONFIG)) |
|
64
|
|
|
|
|
65
|
|
|
|
|
66
|
1 |
|
def delete_conflicts(root=None, config_only=False, force=False): |
|
67
|
|
|
"""Delete all files with conflicted filenames.""" |
|
68
|
1 |
|
root = root or find_root() |
|
69
|
|
|
|
|
70
|
1 |
|
log.info("%s conflicted files...", 'Deleting' if force else 'Displaying') |
|
71
|
1 |
|
pattern = CONFLICT_CONFIG if config_only else CONFLICT_ANY |
|
72
|
1 |
|
log.debug("Pattern: %r", pattern) |
|
73
|
1 |
|
regex = re.compile(pattern) |
|
74
|
1 |
|
count = 0 |
|
75
|
1 |
|
for dirname, _, filenames in os.walk(root): |
|
76
|
1 |
|
for filename in filenames: |
|
77
|
1 |
|
if regex.match(filename): |
|
78
|
1 |
|
count += 1 |
|
79
|
1 |
|
path = os.path.join(dirname, filename) |
|
80
|
1 |
|
if force: |
|
81
|
1 |
|
os.remove(path) |
|
82
|
1 |
|
print(path) |
|
83
|
|
|
|
|
84
|
1 |
|
if count and not force: |
|
85
|
1 |
|
print() |
|
86
|
1 |
|
print("Run again with '--force' to delete these " |
|
87
|
|
|
"{} conflict(s)".format(count)) |
|
88
|
1 |
|
return False |
|
89
|
|
|
else: |
|
90
|
|
|
return True |
|
91
|
|
|
|