path.py
491 lines
| 16.4 KiB
| text/x-python
|
PythonLexer
Brian Granger
|
r2498 | # encoding: utf-8 | ||
""" | ||||
Utilities for path handling. | ||||
""" | ||||
#----------------------------------------------------------------------------- | ||||
Matthias BUSSONNIER
|
r5390 | # Copyright (C) 2008-2011 The IPython Development Team | ||
Brian Granger
|
r2498 | # | ||
# Distributed under the terms of the BSD License. The full license is in | ||||
# the file COPYING, distributed as part of this software. | ||||
#----------------------------------------------------------------------------- | ||||
#----------------------------------------------------------------------------- | ||||
# Imports | ||||
#----------------------------------------------------------------------------- | ||||
import os | ||||
import sys | ||||
MinRK
|
r4475 | import tempfile | ||
Thomas Kluyver
|
r4902 | import warnings | ||
Thomas Kluyver
|
r4177 | from hashlib import md5 | ||
Takafumi Arakaki
|
r8066 | import glob | ||
Brian Granger
|
r2498 | |||
import IPython | ||||
Jörgen Stenarson
|
r7512 | from IPython.testing.skipdoctest import skip_doctest | ||
Fernando Perez
|
r2908 | from IPython.utils.process import system | ||
Brian Granger
|
r2498 | from IPython.utils.importstring import import_item | ||
Thomas Kluyver
|
r4731 | from IPython.utils import py3compat | ||
Brian Granger
|
r2498 | #----------------------------------------------------------------------------- | ||
# Code | ||||
#----------------------------------------------------------------------------- | ||||
Thomas Kluyver
|
r3808 | fs_encoding = sys.getfilesystemencoding() | ||
Brian Granger
|
r2498 | def _get_long_path_name(path): | ||
"""Dummy no-op.""" | ||||
return path | ||||
MinRK
|
r4473 | def _writable_dir(path): | ||
"""Whether `path` is a directory, to which the user has write access.""" | ||||
return os.path.isdir(path) and os.access(path, os.W_OK) | ||||
Brian Granger
|
r2498 | if sys.platform == 'win32': | ||
Jörgen Stenarson
|
r7512 | @skip_doctest | ||
Brian Granger
|
r2498 | def _get_long_path_name(path): | ||
"""Get a long path name (expand ~) on Windows using ctypes. | ||||
Examples | ||||
-------- | ||||
>>> get_long_path_name('c:\\docume~1') | ||||
Jörgen Stenarson
|
r7512 | u'c:\\\\Documents and Settings' | ||
Brian Granger
|
r2498 | |||
""" | ||||
try: | ||||
import ctypes | ||||
except ImportError: | ||||
raise ImportError('you need to have ctypes installed for this to work') | ||||
_GetLongPathName = ctypes.windll.kernel32.GetLongPathNameW | ||||
_GetLongPathName.argtypes = [ctypes.c_wchar_p, ctypes.c_wchar_p, | ||||
ctypes.c_uint ] | ||||
buf = ctypes.create_unicode_buffer(260) | ||||
rv = _GetLongPathName(path, buf, 260) | ||||
if rv == 0 or rv > 260: | ||||
return path | ||||
else: | ||||
return buf.value | ||||
def get_long_path_name(path): | ||||
"""Expand a path into its long form. | ||||
On Windows this expands any ~ in the paths. On other platforms, it is | ||||
a null operation. | ||||
""" | ||||
return _get_long_path_name(path) | ||||
Robert Kern
|
r4696 | def unquote_filename(name, win32=(sys.platform=='win32')): | ||
""" On Windows, remove leading and trailing quotes from filenames. | ||||
""" | ||||
if win32: | ||||
if name.startswith(("'", '"')) and name.endswith(("'", '"')): | ||||
name = name[1:-1] | ||||
return name | ||||
def get_py_filename(name, force_win32=None): | ||||
Brian Granger
|
r2498 | """Return a valid python filename in the current directory. | ||
If the given name is not a file, it adds '.py' and searches again. | ||||
Robert Kern
|
r4688 | Raises IOError with an informative message if the file isn't found. | ||
Bernardo B. Marques
|
r4872 | |||
Robert Kern
|
r4696 | On Windows, apply Windows semantics to the filename. In particular, remove | ||
any quoting that has been applied to it. This option can be forced for | ||||
testing purposes. | ||||
Robert Kern
|
r4688 | """ | ||
Brian Granger
|
r2498 | |||
name = os.path.expanduser(name) | ||||
Robert Kern
|
r4696 | if force_win32 is None: | ||
win32 = (sys.platform == 'win32') | ||||
else: | ||||
win32 = force_win32 | ||||
name = unquote_filename(name, win32=win32) | ||||
Brian Granger
|
r2498 | if not os.path.isfile(name) and not name.endswith('.py'): | ||
name += '.py' | ||||
if os.path.isfile(name): | ||||
return name | ||||
else: | ||||
Bradley M. Froehle
|
r7843 | raise IOError('File `%r` not found.' % name) | ||
Brian Granger
|
r2498 | |||
def filefind(filename, path_dirs=None): | ||||
"""Find a file by looking through a sequence of paths. | ||||
This iterates through a sequence of paths looking for a file and returns | ||||
the full, absolute path of the first occurence of the file. If no set of | ||||
path dirs is given, the filename is tested as is, after running through | ||||
:func:`expandvars` and :func:`expanduser`. Thus a simple call:: | ||||
filefind('myfile.txt') | ||||
will find the file in the current working dir, but:: | ||||
filefind('~/myfile.txt') | ||||
Will find the file in the users home directory. This function does not | ||||
automatically try any paths, such as the cwd or the user's home directory. | ||||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2498 | Parameters | ||
---------- | ||||
filename : str | ||||
The filename to look for. | ||||
path_dirs : str, None or sequence of str | ||||
The sequence of paths to look for the file in. If None, the filename | ||||
need to be absolute or be in the cwd. If a string, the string is | ||||
put into a sequence and the searched. If a sequence, walk through | ||||
each element and join with ``filename``, calling :func:`expandvars` | ||||
and :func:`expanduser` before testing for existence. | ||||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2498 | Returns | ||
------- | ||||
Raises :exc:`IOError` or returns absolute path to file. | ||||
""" | ||||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2498 | # If paths are quoted, abspath gets confused, strip them... | ||
filename = filename.strip('"').strip("'") | ||||
# If the input is an absolute path, just check it exists | ||||
if os.path.isabs(filename) and os.path.isfile(filename): | ||||
return filename | ||||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2498 | if path_dirs is None: | ||
path_dirs = ("",) | ||||
elif isinstance(path_dirs, basestring): | ||||
path_dirs = (path_dirs,) | ||||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2498 | for path in path_dirs: | ||
Jörgen Stenarson
|
r4206 | if path == '.': path = os.getcwdu() | ||
Brian Granger
|
r2498 | testname = expand_path(os.path.join(path, filename)) | ||
if os.path.isfile(testname): | ||||
return os.path.abspath(testname) | ||||
Bernardo B. Marques
|
r4872 | |||
raise IOError("File %r does not exist in any of the search paths: %r" % | ||||
Brian Granger
|
r2498 | (filename, path_dirs) ) | ||
class HomeDirError(Exception): | ||||
pass | ||||
MinRK
|
r5384 | def get_home_dir(require_writable=False): | ||
"""Return the 'home' directory, as a unicode string. | ||||
Brian Granger
|
r2498 | |||
MinRK
|
r5383 | * First, check for frozen env in case of py2exe | ||
MinRK
|
r5384 | * Otherwise, defer to os.path.expanduser('~') | ||
MinRK
|
r5383 | |||
See stdlib docs for how this is determined. | ||||
$HOME is first priority on *ALL* platforms. | ||||
MinRK
|
r5384 | |||
Parameters | ||||
---------- | ||||
require_writable : bool [default: False] | ||||
if True: | ||||
guarantees the return value is a writable directory, otherwise | ||||
raises HomeDirError | ||||
if False: | ||||
The path is resolved, but it is not guaranteed to exist or be writable. | ||||
Brian Granger
|
r2498 | """ | ||
# first, check py2exe distribution root directory for _ipython. | ||||
# This overrides all. Normally does not exist. | ||||
if hasattr(sys, "frozen"): #Is frozen by py2exe | ||||
if '\\library.zip\\' in IPython.__file__.lower():#libraries compressed to zip-file | ||||
root, rest = IPython.__file__.lower().split('library.zip') | ||||
Bernardo B. Marques
|
r4872 | else: | ||
Brian Granger
|
r2498 | root=os.path.join(os.path.split(IPython.__file__)[0],"../../") | ||
root=os.path.abspath(root).rstrip('\\') | ||||
MinRK
|
r4474 | if _writable_dir(os.path.join(root, '_ipython')): | ||
Brian Granger
|
r2498 | os.environ["IPYKITROOT"] = root | ||
Thomas Kluyver
|
r4731 | return py3compat.cast_unicode(root, fs_encoding) | ||
MinRK
|
r5383 | |||
homedir = os.path.expanduser('~') | ||||
Paul Ivanov
|
r6112 | # Next line will make things work even when /home/ is a symlink to | ||
# /usr/home as it is on FreeBSD, for example | ||||
homedir = os.path.realpath(homedir) | ||||
MinRK
|
r5385 | |||
if not _writable_dir(homedir) and os.name == 'nt': | ||||
# expanduser failed, use the registry to get the 'My Documents' folder. | ||||
Brian Granger
|
r2498 | try: | ||
import _winreg as wreg | ||||
key = wreg.OpenKey( | ||||
wreg.HKEY_CURRENT_USER, | ||||
"Software\Microsoft\Windows\CurrentVersion\Explorer\Shell Folders" | ||||
) | ||||
homedir = wreg.QueryValueEx(key,'Personal')[0] | ||||
key.Close() | ||||
except: | ||||
pass | ||||
MinRK
|
r5385 | |||
MinRK
|
r5384 | if (not require_writable) or _writable_dir(homedir): | ||
MinRK
|
r5383 | return py3compat.cast_unicode(homedir, fs_encoding) | ||
Brian Granger
|
r2498 | else: | ||
MinRK
|
r5385 | raise HomeDirError('%s is not a writable dir, ' | ||
'set $HOME environment variable to override' % homedir) | ||||
Brian Granger
|
r2498 | |||
MinRK
|
r3347 | def get_xdg_dir(): | ||
"""Return the XDG_CONFIG_HOME, if it is defined and exists, else None. | ||||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r7086 | This is only for non-OS X posix (Linux,Unix,etc.) systems. | ||
MinRK
|
r3347 | """ | ||
env = os.environ | ||||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r7086 | if os.name == 'posix' and sys.platform != 'darwin': | ||
# Linux, Unix, AIX, etc. | ||||
MinRK
|
r5384 | # use ~/.config if empty OR not set | ||
MinRK
|
r3347 | xdg = env.get("XDG_CONFIG_HOME", None) or os.path.join(get_home_dir(), '.config') | ||
MinRK
|
r4474 | if xdg and _writable_dir(xdg): | ||
Thomas Kluyver
|
r4731 | return py3compat.cast_unicode(xdg, fs_encoding) | ||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r3347 | return None | ||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2498 | |||
def get_ipython_dir(): | ||||
"""Get the IPython directory for this platform and user. | ||||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2498 | This uses the logic in `get_home_dir` to find the home directory | ||
Thomas Kluyver
|
r4731 | and then adds .ipython to the end of the path. | ||
Brian Granger
|
r2498 | """ | ||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r3347 | env = os.environ | ||
pjoin = os.path.join | ||||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2498 | ipdir_def = '.ipython' | ||
MinRK
|
r3347 | xdg_def = 'ipython' | ||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2498 | home_dir = get_home_dir() | ||
MinRK
|
r3347 | xdg_dir = get_xdg_dir() | ||
MinRK
|
r5384 | |||
Brian Granger
|
r2505 | # import pdb; pdb.set_trace() # dbg | ||
Bradley M. Froehle
|
r6699 | if 'IPYTHON_DIR' in env: | ||
warnings.warn('The environment variable IPYTHON_DIR is deprecated. ' | ||||
'Please use IPYTHONDIR instead.') | ||||
Bradley M. Froehle
|
r6697 | ipdir = env.get('IPYTHONDIR', env.get('IPYTHON_DIR', None)) | ||
MinRK
|
r3347 | if ipdir is None: | ||
# not set explicitly, use XDG_CONFIG_HOME or HOME | ||||
home_ipdir = pjoin(home_dir, ipdir_def) | ||||
if xdg_dir: | ||||
# use XDG, as long as the user isn't already | ||||
# using $HOME/.ipython and *not* XDG/ipython | ||||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r3347 | xdg_ipdir = pjoin(xdg_dir, xdg_def) | ||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r4474 | if _writable_dir(xdg_ipdir) or not _writable_dir(home_ipdir): | ||
MinRK
|
r3347 | ipdir = xdg_ipdir | ||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r3347 | if ipdir is None: | ||
# not using XDG | ||||
ipdir = home_ipdir | ||||
Thomas Kluyver
|
r3808 | |||
MinRK
|
r3896 | ipdir = os.path.normpath(os.path.expanduser(ipdir)) | ||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r4475 | if os.path.exists(ipdir) and not _writable_dir(ipdir): | ||
# ipdir exists, but is not writable | ||||
Thomas Kluyver
|
r4902 | warnings.warn("IPython dir '%s' is not a writable location," | ||
MinRK
|
r4475 | " using a temp directory."%ipdir) | ||
ipdir = tempfile.mkdtemp() | ||||
elif not os.path.exists(ipdir): | ||||
parent = ipdir.rsplit(os.path.sep, 1)[0] | ||||
if not _writable_dir(parent): | ||||
# ipdir does not exist and parent isn't writable | ||||
Thomas Kluyver
|
r4902 | warnings.warn("IPython parent '%s' is not a writable location," | ||
MinRK
|
r4475 | " using a temp directory."%parent) | ||
ipdir = tempfile.mkdtemp() | ||||
MinRK
|
r3896 | |||
Thomas Kluyver
|
r4731 | return py3compat.cast_unicode(ipdir, fs_encoding) | ||
Brian Granger
|
r2498 | |||
def get_ipython_package_dir(): | ||||
"""Get the base directory where IPython itself is installed.""" | ||||
ipdir = os.path.dirname(IPython.__file__) | ||||
Thomas Kluyver
|
r4731 | return py3compat.cast_unicode(ipdir, fs_encoding) | ||
Brian Granger
|
r2498 | |||
def get_ipython_module_path(module_str): | ||||
"""Find the path to an IPython module in this version of IPython. | ||||
This will always find the version of the module that is in this importable | ||||
IPython package. This will always return the path to the ``.py`` | ||||
version of the module. | ||||
""" | ||||
if module_str == 'IPython': | ||||
return os.path.join(get_ipython_package_dir(), '__init__.py') | ||||
mod = import_item(module_str) | ||||
the_path = mod.__file__.replace('.pyc', '.py') | ||||
the_path = the_path.replace('.pyo', '.py') | ||||
Thomas Kluyver
|
r4731 | return py3compat.cast_unicode(the_path, fs_encoding) | ||
Brian Granger
|
r2498 | |||
Thomas Kluyver
|
r4988 | def locate_profile(profile='default'): | ||
"""Find the path to the folder associated with a given profile. | ||||
Bradley M. Froehle
|
r6696 | I.e. find $IPYTHONDIR/profile_whatever. | ||
Thomas Kluyver
|
r4988 | """ | ||
from IPython.core.profiledir import ProfileDir, ProfileDirError | ||||
try: | ||||
pd = ProfileDir.find_profile_dir_by_name(get_ipython_dir(), profile) | ||||
except ProfileDirError: | ||||
# IOError makes more sense when people are expecting a path | ||||
raise IOError("Couldn't find profile %r" % profile) | ||||
return pd.location | ||||
Brian Granger
|
r2498 | |||
def expand_path(s): | ||||
"""Expand $VARS and ~names in a string, like a shell | ||||
:Examples: | ||||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2498 | In [2]: os.environ['FOO']='test' | ||
In [3]: expand_path('variable FOO is $FOO') | ||||
Out[3]: 'variable FOO is test' | ||||
""" | ||||
# This is a pretty subtle hack. When expand user is given a UNC path | ||||
# on Windows (\\server\share$\%username%), os.path.expandvars, removes | ||||
# the $ to get (\\server\share\%username%). I think it considered $ | ||||
# alone an empty var. But, we need the $ to remains there (it indicates | ||||
# a hidden share). | ||||
if os.name=='nt': | ||||
s = s.replace('$\\', 'IPYTHON_TEMP') | ||||
s = os.path.expandvars(os.path.expanduser(s)) | ||||
if os.name=='nt': | ||||
s = s.replace('IPYTHON_TEMP', '$\\') | ||||
return s | ||||
Takafumi Arakaki
|
r8119 | def unescape_glob(string): | ||
"""Unescape glob pattern in `string`.""" | ||||
Takafumi Arakaki
|
r8122 | def unescape(s): | ||
for pattern in '*[]!?': | ||||
s = s.replace(r'\{0}'.format(pattern), pattern) | ||||
return s | ||||
return '\\'.join(map(unescape, string.split('\\\\'))) | ||||
Takafumi Arakaki
|
r8119 | |||
Takafumi Arakaki
|
r8067 | def shellglob(args): | ||
Takafumi Arakaki
|
r8014 | """ | ||
Do glob expansion for each element in `args` and return a flattened list. | ||||
Unmatched glob pattern will remain as-is in the returned list. | ||||
""" | ||||
expanded = [] | ||||
for a in args: | ||||
Takafumi Arakaki
|
r8119 | expanded.extend(glob.glob(a) or [unescape_glob(a)]) | ||
Takafumi Arakaki
|
r8014 | return expanded | ||
Brian Granger
|
r2498 | def target_outdated(target,deps): | ||
"""Determine whether a target is out of date. | ||||
target_outdated(target,deps) -> 1/0 | ||||
deps: list of filenames which MUST exist. | ||||
target: single filename which may or may not exist. | ||||
If target doesn't exist or is older than any file listed in deps, return | ||||
true, otherwise return false. | ||||
""" | ||||
try: | ||||
target_time = os.path.getmtime(target) | ||||
except os.error: | ||||
return 1 | ||||
for dep in deps: | ||||
dep_time = os.path.getmtime(dep) | ||||
if dep_time > target_time: | ||||
#print "For target",target,"Dep failed:",dep # dbg | ||||
#print "times (dep,tar):",dep_time,target_time # dbg | ||||
return 1 | ||||
return 0 | ||||
def target_update(target,deps,cmd): | ||||
"""Update a target with a given command given a list of dependencies. | ||||
target_update(target,deps,cmd) -> runs cmd if target is outdated. | ||||
This is just a wrapper around target_outdated() which calls the given | ||||
command if target is outdated.""" | ||||
if target_outdated(target,deps): | ||||
Fernando Perez
|
r2908 | system(cmd) | ||
Brian Granger
|
r2498 | |||
Thomas Kluyver
|
r4177 | def filehash(path): | ||
"""Make an MD5 hash of a file, ignoring any differences in line | ||||
ending characters.""" | ||||
with open(path, "rU") as f: | ||||
Thomas Kluyver
|
r4731 | return md5(py3compat.str_to_bytes(f.read())).hexdigest() | ||
Thomas Kluyver
|
r4177 | |||
# If the config is unmodified from the default, we'll just delete it. | ||||
# These are consistent for 0.10.x, thankfully. We're not going to worry about | ||||
# older versions. | ||||
old_config_md5 = {'ipy_user_conf.py': 'fc108bedff4b9a00f91fa0a5999140d3', | ||||
'ipythonrc': '12a68954f3403eea2eec09dc8fe5a9b5'} | ||||
MinRK
|
r3836 | def check_for_old_config(ipython_dir=None): | ||
"""Check for old config files, and present a warning if they exist. | ||||
A link to the docs of the new config is included in the message. | ||||
This should mitigate confusion with the transition to the new | ||||
config system in 0.11. | ||||
""" | ||||
if ipython_dir is None: | ||||
ipython_dir = get_ipython_dir() | ||||
MinRK
|
r4024 | old_configs = ['ipy_user_conf.py', 'ipythonrc', 'ipython_config.py'] | ||
MinRK
|
r4037 | warned = False | ||
MinRK
|
r3836 | for cfg in old_configs: | ||
f = os.path.join(ipython_dir, cfg) | ||||
if os.path.exists(f): | ||||
Thomas Kluyver
|
r4178 | if filehash(f) == old_config_md5.get(cfg, ''): | ||
Thomas Kluyver
|
r4177 | os.unlink(f) | ||
else: | ||||
Thomas Kluyver
|
r4902 | warnings.warn("Found old IPython config file %r (modified by user)"%f) | ||
Thomas Kluyver
|
r4177 | warned = True | ||
Bernardo B. Marques
|
r4872 | |||
MinRK
|
r4037 | if warned: | ||
Thomas Kluyver
|
r4902 | warnings.warn(""" | ||
Thomas Kluyver
|
r4180 | The IPython configuration system has changed as of 0.11, and these files will | ||
be ignored. See http://ipython.github.com/ipython-doc/dev/config for details | ||||
of the new config system. | ||||
To start configuring IPython, do `ipython profile create`, and edit | ||||
`ipython_config.py` in <ipython_dir>/profile_default. | ||||
If you need to leave the old config files in place for an older version of | ||||
Paul Ivanov
|
r4201 | IPython and want to suppress this warning message, set | ||
`c.InteractiveShellApp.ignore_old_config=True` in the new config.""") | ||||
MinRK
|
r3837 | |||
MinRK
|
r4968 | def get_security_file(filename, profile='default'): | ||
"""Return the absolute path of a security file given by filename and profile | ||||
This allows users and developers to find security files without | ||||
knowledge of the IPython directory structure. The search path | ||||
will be ['.', profile.security_dir] | ||||
Parameters | ||||
---------- | ||||
filename : str | ||||
The file to be found. If it is passed as an absolute path, it will | ||||
simply be returned. | ||||
profile : str [default: 'default'] | ||||
The name of the profile to search. Leaving this unspecified | ||||
The file to be found. If it is passed as an absolute path, fname will | ||||
simply be returned. | ||||
Returns | ||||
------- | ||||
Raises :exc:`IOError` if file not found or returns absolute path to file. | ||||
""" | ||||
# import here, because profiledir also imports from utils.path | ||||
from IPython.core.profiledir import ProfileDir | ||||
try: | ||||
pd = ProfileDir.find_profile_dir_by_name(get_ipython_dir(), profile) | ||||
except Exception: | ||||
# will raise ProfileDirError if no such profile | ||||
raise IOError("Profile %r not found") | ||||
return filefind(filename, ['.', pd.security_dir]) | ||||