Как получить / задать логический путь к каталогу в python
Вопрос
В python возможно ли получить или задать логический каталог (в отличие от абсолютного).
Например, если у меня есть:
/real/path/to/dir
и у меня есть
/linked/path/to/dir
связан с тем же каталогом.
использование os.getcwd и os.chdir всегда будет использовать абсолютный путь
>>> import os
>>> os.chdir('/linked/path/to/dir')
>>> print os.getcwd()
/real/path/to/dir
Единственный способ, который я нашел, чтобы вообще обойти это, - запустить 'pwd' в другом процессе и прочитать выходные данные.Однако это работает только до тех пор, пока вы не вызовете os.chdir в первый раз.
Решение
Базовая операционная система / оболочка сообщает реальные пути к python.
Таким образом, на самом деле нет никакого способа обойти это, поскольку os.getcwd()
это обернутый вызов библиотеки C getcwd()
функция.
Есть несколько обходных путей в духе того, который вы уже знаете и который запускается pwd
.
Другой способ предполагал бы использование os.environ['PWD']
.Если эта переменная среды установлена, вы можете сделать некоторые getcwd
функция, которая уважает это.
Приведенное ниже решение сочетает в себе оба:
import os
from subprocess import Popen, PIPE
class CwdKeeper(object):
def __init__(self):
self._cwd = os.environ.get("PWD")
if self._cwd is None: # no environment. fall back to calling pwd on shell
self._cwd = Popen('pwd', stdout=PIPE).communicate()[0].strip()
self._os_getcwd = os.getcwd
self._os_chdir = os.chdir
def chdir(self, path):
if not self._cwd:
return self._os_chdir(path)
p = os.path.normpath(os.path.join(self._cwd, path))
result = self._os_chdir(p)
self._cwd = p
os.environ["PWD"] = p
return result
def getcwd(self):
if not self._cwd:
return self._os_getcwd()
return self._cwd
cwd = CwdKeeper()
print cwd.getcwd()
# use only cwd.chdir and cwd.getcwd from now on.
# monkeypatch os if you want:
os.chdir = cwd.chdir
os.getcwd = cwd.getcwd
# now you can use os.chdir and os.getcwd as normal.
Другие советы
Для меня это тоже помогает:
import os
os.popen('pwd').read().strip('\n')
Вот демонстрация в оболочке python:
>>> import os
>>> os.popen('pwd').read()
'/home/projteam/staging/site/proj\n'
>>> os.popen('pwd').read().strip('\n')
'/home/projteam/staging/site/proj'
>>> # Also works if PWD env var is set
>>> os.getenv('PWD')
'/home/projteam/staging/site/proj'
>>> # This gets actual path, not symlinked path
>>> import subprocess
>>> p = subprocess.Popen('pwd', stdout=subprocess.PIPE)
>>> p.communicate()[0] # returns non-symlink path
'/home/projteam/staging/deploys/20150114-141114/site/proj\n'
Получение переменной окружения PWD не всегда работало у меня, поэтому я использую метод popen.Ваше здоровье!