-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathrelpath.py
87 lines (75 loc) · 2.52 KB
/
relpath.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
#----------------------------------------------------------------------
# Name: relpath.py
# Purpose:
#
# Author: Riaan Booysen
#
# Created: 1999
# RCS-ID: $Id$
# Copyright: (c) 1999 - 2003 Riaan Booysen
# Licence: GPL
#----------------------------------------------------------------------
##b = 'c:\\a\\b\\f\\d'
##d = 'c:\\a\\b\\f\\h.txt'
##d = 'c:\\a\\b\\h\\i\\j.txt'
##e = 'd:\\z\\x\\y\\j\\k.txt'
import os
def splitpath(apath):
""" Splits a path into a list of directory names """
path_list = []
drive, apath = os.path.splitdrive(apath)
head, tail = os.path.split(apath)
while 1:
if tail:
path_list.insert(0, tail)
newhead, tail = os.path.split(head)
if newhead == head:
break
else:
head = newhead
if drive:
path_list.insert(0, drive)
return path_list
def relpath(base, comp):
""" Return a path to file comp relative to path base. """
protsplitbase = base.split('://')
if len(protsplitbase) == 1:
baseprot, nbase = 'file', protsplitbase[0]
elif len(protsplitbase) == 2:
baseprot, nbase = protsplitbase
elif len(protsplitbase) == 3:
baseprot, nbase, zipentry = protsplitbase
else:
raise Exception, 'Unhandled path %s'%`protsplitbase`
protsplitcomp = comp.split('://')
if len(protsplitcomp) == 1:
compprot, ncomp = 'file', protsplitcomp[0]
elif len(protsplitcomp) == 2:
compprot, ncomp = protsplitcomp
elif len(protsplitcomp) == 3:
compprot, ncomp, zipentry = protsplitcomp
else:
raise Exception, 'Unhandled path %s'%`protsplitcomp`
if baseprot != compprot:
return comp
base_drive, base_path = os.path.splitdrive(nbase)
comp_drive, comp_path = os.path.splitdrive(ncomp)
base_path_list = splitpath(base_path)
comp_path_list = splitpath(comp_path)
if base_drive != comp_drive:
return comp
# relative path defaults to the list of files with
# a greater index then the entire base
rel_path = comp_path_list[len(base_path_list):]
# find the first directory for which the 2 paths differ
found = -1
idx = 0
for idx in range(len(base_path_list)):
if base_path_list[idx].lower() != comp_path_list[idx].lower():
rel_path = comp_path_list[idx:]
found = 0
break
for cnt in range(max(len(base_path_list) - idx + found, 0)):
rel_path.insert(0, os.pardir)
return os.path.join(*rel_path)
#print relpath(b, d)