##// END OF EJS Templates
validate that ESC_PAREN ('/') is followed by a callable name and not empty
validate that ESC_PAREN ('/') is followed by a callable name and not empty

File last commit:

r28822:01e08c9c
r28935:e8c98709
Show More
profiledir.py
244 lines | 8.3 KiB | text/x-python | PythonLexer
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024 # encoding: utf-8
MinRK
add utils.path.ensure_dir_exists...
r16486 """An object for managing IPython profile directories."""
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024
MinRK
add utils.path.ensure_dir_exists...
r16486 # Copyright (c) IPython Development Team.
# Distributed under the terms of the Modified BSD License.
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024
import os
import shutil
Paul Ivanov
fix missing import
r12266 import errno
Matthias Bussonnier
Use Pathlib here and there....
r25972 from pathlib import Path
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024
Min RK
update dependency imports...
r21253 from traitlets.config.configurable import LoggingConfigurable
Srinivas Reddy Thatiparthy
Change absolute imports to relative imports to facilitate processes embedding kernel or debugger
r25227 from ..paths import get_ipython_package_dir
from ..utils.path import expand_path, ensure_dir_exists
Min RK
adopt traitlets 4.2 API...
r22340 from traitlets import Unicode, Bool, observe
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024
M Bussonnier
Only copy files in startup dir if we just created it....
r28821 from typing import Optional
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024 #-----------------------------------------------------------------------------
# Module errors
#-----------------------------------------------------------------------------
class ProfileDirError(Exception):
pass
#-----------------------------------------------------------------------------
# Class for managing profile directories
#-----------------------------------------------------------------------------
MinRK
Prevents crash on some systems where chmod fails (e.g. sshfs on Windows)....
r5197 class ProfileDir(LoggingConfigurable):
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024 """An object to manage the profile directory and its resources.
The profile directory is used by all IPython applications, to manage
configuration, logging and security.
This object knows how to find, create and manage these directories. This
should be used by any code that wants to handle profiles.
"""
security_dir_name = Unicode('security')
log_dir_name = Unicode('log')
MinRK
add startup_dir to profiles...
r5246 startup_dir_name = Unicode('startup')
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024 pid_dir_name = Unicode('pid')
MinRK
create static/custom in skeleton profiles
r12813 static_dir_name = Unicode('static')
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024 security_dir = Unicode(u'')
log_dir = Unicode(u'')
MinRK
add startup_dir to profiles...
r5246 startup_dir = Unicode(u'')
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024 pid_dir = Unicode(u'')
MinRK
create static/custom in skeleton profiles
r12813 static_dir = Unicode(u'')
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024
Min RK
adopt traitlets 4.2 API...
r22340 location = Unicode(u'',
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024 help="""Set the profile location directly. This overrides the logic used by the
`profile` option.""",
Min RK
adopt traitlets 4.2 API...
r22340 ).tag(config=True)
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024
_location_isset = Bool(False) # flag for detecting multiply set location
Min RK
adopt traitlets 4.2 API...
r22340 @observe('location')
def _location_changed(self, change):
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024 if self._location_isset:
raise RuntimeError("Cannot set profile location more than once.")
self._location_isset = True
Min RK
adopt traitlets 4.2 API...
r22340 new = change['new']
MinRK
add utils.path.ensure_dir_exists...
r16486 ensure_dir_exists(new)
James Porter
fix race condition in profiledir creation.
r13981
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024 # ensure config files exist:
self.security_dir = os.path.join(new, self.security_dir_name)
self.log_dir = os.path.join(new, self.log_dir_name)
MinRK
add startup_dir to profiles...
r5246 self.startup_dir = os.path.join(new, self.startup_dir_name)
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024 self.pid_dir = os.path.join(new, self.pid_dir_name)
MinRK
create static/custom in skeleton profiles
r12813 self.static_dir = os.path.join(new, self.static_dir_name)
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024 self.check_dirs()
M Bussonnier
Only copy files in startup dir if we just created it....
r28821
def _mkdir(self, path: str, mode: Optional[int] = None) -> bool:
MinRK
avoid race condition in profile creation...
r10351 """ensure a directory exists at a given path
James Porter
fix race condition in profiledir creation.
r13981
MinRK
avoid race condition in profile creation...
r10351 This is a version of os.mkdir, with the following differences:
James Porter
fix race condition in profiledir creation.
r13981
M Bussonnier
Apply suggestions from code review...
r28822 - returns whether the directory has been created or not.
MinRK
avoid race condition in profile creation...
r10351 - ignores EEXIST, protecting against race conditions where
the dir may have been created in between the check and
the creation
- sets permissions if requested and the dir already exists
M Bussonnier
Only copy files in startup dir if we just created it....
r28821
Parameters
----------
path: str
path of the dir to create
mode: int
see `mode` of `os.mkdir`
Returns
-------
bool:
returns True if it created the directory, False otherwise
MinRK
avoid race condition in profile creation...
r10351 """
M Bussonnier
Only copy files in startup dir if we just created it....
r28821
MinRK
avoid race condition in profile creation...
r10351 if os.path.exists(path):
if mode and os.stat(path).st_mode != mode:
try:
os.chmod(path, mode)
except OSError:
Pierre Gerold
Same in profileapp.py , profiledir.py, shellapp.py
r21888 self.log.warning(
MinRK
avoid race condition in profile creation...
r10351 "Could not set permissions on %s",
path
)
return False
try:
if mode:
os.mkdir(path, mode)
else:
os.mkdir(path)
except OSError as e:
if e.errno == errno.EEXIST:
return False
else:
raise
James Porter
fix race condition in profiledir creation.
r13981
MinRK
avoid race condition in profile creation...
r10351 return True
Min RK
adopt traitlets 4.2 API...
r22340
@observe('log_dir')
def check_log_dir(self, change=None):
MinRK
avoid race condition in profile creation...
r10351 self._mkdir(self.log_dir)
Min RK
adopt traitlets 4.2 API...
r22340
@observe('startup_dir')
def check_startup_dir(self, change=None):
M Bussonnier
Only copy files in startup dir if we just created it....
r28821 if self._mkdir(self.startup_dir):
readme = os.path.join(self.startup_dir, "README")
src = os.path.join(
get_ipython_package_dir(), "core", "profile", "README_STARTUP"
)
M Bussonnier
Apply suggestions from code review...
r28822 if os.path.exists(src):
if not os.path.exists(readme):
shutil.copy(src, readme)
else:
M Bussonnier
Only copy files in startup dir if we just created it....
r28821 self.log.warning(
"Could not copy README_STARTUP to startup dir. Source file %s does not exist.",
src,
)
Min RK
adopt traitlets 4.2 API...
r22340 @observe('security_dir')
def check_security_dir(self, change=None):
MinRK
avoid race condition in profile creation...
r10351 self._mkdir(self.security_dir, 0o40700)
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024
Min RK
adopt traitlets 4.2 API...
r22340 @observe('pid_dir')
def check_pid_dir(self, change=None):
MinRK
avoid race condition in profile creation...
r10351 self._mkdir(self.pid_dir, 0o40700)
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024
def check_dirs(self):
self.check_security_dir()
self.check_log_dir()
self.check_pid_dir()
MinRK
add README to startup dir
r5247 self.check_startup_dir()
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024
Matthias Bussonnier
Use Pathlib here and there....
r25972 def copy_config_file(self, config_file: str, path: Path, overwrite=False) -> bool:
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024 """Copy a default config file into the active profile directory.
Min RK
update dependency imports...
r21253 Default configuration files are kept in :mod:`IPython.core.profile`.
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024 This function moves these from that location to the working profile
directory.
"""
Matthias Bussonnier
Use Pathlib here and there....
r25972 dst = Path(os.path.join(self.location, config_file))
if dst.exists() and not overwrite:
MinRK
load bundled profiles without having to use 'profile create' or '--init'...
r4122 return False
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024 if path is None:
Min RK
remove bundled profiles...
r20870 path = os.path.join(get_ipython_package_dir(), u'core', u'profile', u'default')
Matthias Bussonnier
Use Pathlib here and there....
r25972 assert isinstance(path, Path)
src = path / config_file
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024 shutil.copy(src, dst)
MinRK
load bundled profiles without having to use 'profile create' or '--init'...
r4122 return True
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024
@classmethod
def create_profile_dir(cls, profile_dir, config=None):
"""Create a new profile directory given a full path.
Parameters
----------
profile_dir : str
The full path to the profile directory. If it does exist, it will
be used. If not, it will be created.
"""
return cls(location=profile_dir, config=config)
@classmethod
def create_profile_dir_by_name(cls, path, name=u'default', config=None):
"""Create a profile dir by profile name and path.
Parameters
----------
path : unicode
The path (directory) to put the profile directory in.
name : unicode
The name of the profile. The name of the profile directory will
be "profile_<profile>".
"""
if not os.path.isdir(path):
raise ProfileDirError('Directory not found: %s' % path)
profile_dir = os.path.join(path, u'profile_' + name)
return cls(location=profile_dir, config=config)
@classmethod
def find_profile_dir_by_name(cls, ipython_dir, name=u'default', config=None):
"""Find an existing profile dir by profile name, return its ProfileDir.
This searches through a sequence of paths for a profile dir. If it
is not found, a :class:`ProfileDirError` exception will be raised.
The search path algorithm is:
Matthias Bussonnier
FIX CVE-2022-21699...
r27464 1. ``os.getcwd()`` # removed for security reason.
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024 2. ``ipython_dir``
Parameters
----------
ipython_dir : unicode or str
The IPython directory to use.
name : unicode or str
The name of the profile. The name of the profile directory
will be "profile_<profile>".
"""
dirname = u'profile_' + name
Matthias Bussonnier
FIX CVE-2022-21699...
r27464 paths = [ipython_dir]
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024 for p in paths:
profile_dir = os.path.join(p, dirname)
if os.path.isdir(profile_dir):
return cls(location=profile_dir, config=config)
else:
raise ProfileDirError('Profile directory not found in paths: %s' % dirname)
@classmethod
def find_profile_dir(cls, profile_dir, config=None):
"""Find/create a profile dir and return its ProfileDir.
This will create the profile directory if it doesn't exist.
Parameters
----------
profile_dir : unicode or str
Matthias Bussonnier
Remove old unused class...
r21439 The path of the profile directory.
MinRK
move ipcluster create|list to `ipython profile create|list`...
r4024 """
profile_dir = expand_path(profile_dir)
if not os.path.isdir(profile_dir):
raise ProfileDirError('Profile directory not found: %s' % profile_dir)
return cls(location=profile_dir, config=config)