- What are they?
- How do I create one?
- How do they work?
- Where do they live?
- Tips
- Related tools
- Reference
???
Hi there! First, a bit about me. I'm a developer at EnergySage. We're a startup in downtown Boston, and use Django to power the world's first and largest online marketplace for solar panel installation.
- Take your time
- Look ahead
- Breathe
TODO
- s/Python version/Python interpreter/
- Run commands on Ubuntu?
- Better typography
- Quotes
- Personal links
name: assumptions
You've got Python 3 installed on macOS or Linux
You're somewhat familiar with a Unix-like command line
You've used pip install
???
- Before we dive in, let's take a moment to review the assumptions I made when putting together these slides
- Doesn't mean this won't be useful, I just won't be covering how to get to this point
- Not exclusive to Python 3; much of this applies to Python 2
- I'm not a Windows user, but the references provide instructions
- Installing Python is non-trivial; we could do a whole talk (hint). See the "Installing" slide.
- I'll probably use "venv" or "virtualenv" interchangeably with "virtual environment"
- Show of hands?
name: what
Isolated workspaces for Python projects
???
- High level, I like to think of them like this
- When I start working on a new project, I usually create one
--
A place to install and use packages independently from other Python environments
???
- More concretely
- Let's define a couple of those terms
- An isolated copy of Python?
--
"packages" == "distribution packages"
- Libraries, frameworks, standalone apps, etc.
- Installed with
pip
intosite-packages
directory
???
- "packages" is overloaded
- Python Packaging Authority
- requests, pandas, Django, Flask, jupyter, Scrapy
site-packages
inside a Python environment --
"other Python environments"
- OS-installed Python, aka "system Python"
- User-installed Python, e.g. from python.org or Anaconda
- Other virtual environments
???
- An installation of Python
Avoid conflicting package dependencies
???
- At work, we use Django 1.8, but I maintain my band's website uses 1.11
- Allows me to develop both on the same laptop
- System Python might rely on an older version of a package, and you want to use the newer one
- Installing the newer version might break OS tools that depend on the old package
- Keep system Python clean
--
Use different versions of Python for different projects
???
- At work, we use Python 2.7, my band's website uses Python 3.6
- venvs are tied to a specific Python executable, meaning a specific version
--
Install packages without sudo pip install
???
- Might see instructions online for installing to system Python
- But we don't want to touch system Python
- Can't use
sudo
on shared hosting - venvs usually live in your home directory
- Install whatever you want
name: create
A minimal example:
???
- Python 3 makes this easy
- Let's say I want to play with a web API; I'll want to use
requests
.
--
$ python3 -m venv api_venv
???
$
indicates a new command line; the command starts after it- Ask Python 3 to run the
venv
module to create a virtualenv - Creates a directory; we'll take a look inside of that later
--
$ source api_venv/bin/activate
???
- We need to activate the venv to use it
- We'll see what this does later
--
(api_venv)$ pip install requests
# ... install output ...
???
- Now I'm inside the venv, as we can see from the prompt
- It's safe to install packages
--
(api_venv)$ python
Python 3.6.2 (default, Jul 17 2017, 16:44:45)
>>> import requests
# Play with a web API...
>>> ^D
???
- We'll see why
python
works later
--
(api_venv)$ deactivate
--
$
???
- Now I'm outside the venv
--
What just happened?
???
- Let's look at our shell environment before and after we ran
activate
$ which python3 && python3 --version
/usr/local/bin/python3
Python 3.6.2
???
- Ask Python 3 where it lives, and what version it is
- I installed this with Homebrew
--
$ python3 -m pip list
pip (9.0.1)
setuptools (32.2.0)
wheel (0.29.0)
???
- Ask Python what packages are installed
- Using
pip
with a known version of Python
(api_venv)$ which python && python --version
/Users/brian/Code/api_venv/bin/python
Python 3.6.2
???
- Same version, but now it lives in our venv directory
venv
lets us usepython
instead ofpython3
--
(api_venv)$ pip list
certifi (2017.7.27.1)
chardet (3.0.4)
idna (2.5)
pip (9.0.1)
requests (2.18.3)
setuptools (28.8.0)
urllib3 (1.22)
???
requests
and its dependencies- Clearly different environments for running Python
- It's safe to use
pip
inside a venv
name: how
Let's go back in time, and into the weeds...
layout: true
$ which python && python --version
/usr/bin/python
Python 2.7.10
???
- First, let's look at system Python
- This is the version that comes with macOS
--
$ python -m pip list
altgraph (0.10.2)
bdist-mpkg (0.5.0)
bonjour-py (0.3)
macholib (1.5.1)
matplotlib (1.3.1)
modulegraph (0.10.4)
numpy (1.8.0rc1)
pip (9.0.1)
py2app (0.7.3)
pyobjc-core (2.5.1)
pyobjc-framework-Accounts (2.5.1)
pyobjc-framework-AddressBook (2.5.1)
# many more...
???
- So many packages that we don't need or want
$ python -m pip install --user virtualenv
???
- venvs are not part of the language
- First version of
virtualenv
on PyPI on 9/14/2007 - More complicated than
pip install
, but it's the safest cross-platform command
--
$ python -m virtualenv api_virtualenv
New python executable in /Users/brian/Code/api_virtualenv/bin/python
Installing setuptools, pip, wheel...done.
???
- Similar to
python3 -m venv
, create a new virtualenv - Unlike
venv
, output gives us a hint about what's going on
--
Or
$ virtualenv api_virtualenv
???
- Commonly documented, does the same thing
--
But it's not clear which version of Python we're using
???
- Specifying a different version requires a command line argument
- For clarity and consistency, use
python -m virtualenv
classes: small
.small[
api_virtualenv
├── bin
*│ ├── activate
│ ├── pip
│ ├── pip2
*│ ├── python
│ ├── python2 -> python
│ # more executables and activate scripts
├── include
│ └── python2.7 -> /System/Library/Frameworks/Python.framework/Versions/2.7/...
├── lib
│ └── python2.7
│ ├── distutils
*│ ├── site-packages
│ │ ├── pip
│ │ ├── setuptools
│ │ ├── wheel
│ │ # more packaging resources
│ ├── no-global-site-packages.txt
│ ├── orig-prefix.txt
│ ├── site.py
│ ├── os.py -> /System/Library/Frameworks/Python.framework/Versions/2.7/...
│ ├── posixpath.py -> /System/Library/Frameworks/Python.framework/Versions/2.7/...
│ ├── re.py -> /System/Library/Frameworks/Python.framework/Versions/2.7/...
│ # many more symbolic links
]
???
- Let's see what's inside our virtualenv
- Abbreviated output of
tree -L 4 --dirsfirst -I '*.pyc|*.dist-info' api_virtualenv
- All this, and I haven't installed any packages!
- Isolated
site-packages
is what we want; that's where pip installs packages - None of the system packages
- Adds
activate
scripts - What does all the other stuff in
lib
do?
Tells Python to behave like it's installed in the virtualenv directory
--
$ python -c "import sys; print(sys.prefix)"
/System/Library/Frameworks/Python.framework/Versions/2.7
???
sys.prefix
== "Where does system Python think it's installed?"- Determined by moving up from location of executable and looking for specific files
- https://carljm.github.io/pipvirtualenv-preso/#10
--
$ api_virtualenv/bin/python -c "import sys; print(sys.prefix)"
/Users/brian/Code/api_virtualenv
--
$ api_virtualenv/bin/python -m pip list
pip (9.0.1)
setuptools (36.2.7)
wheel (0.29.0)
???
- Sure enough, we're not using the system
site-packages
- But now we don't have a standard library; batteries not included!
Creates hacked copy of site.py
to use standard library from original sys.prefix
$ cat api_virtualenv/lib/python2.7/orig-prefix.txt
/System/Library/Frameworks/Python.framework/Versions/2.7
???
site.py
tells Python where to look for packages- Load the standard library from over here
- https://carljm.github.io/pipvirtualenv-preso/#20
- But
site.py
imports from standard library!
--
Creates symbolic links to standard library modules imported by site.py
???
- I know we're in the weeds, but I think this is really interesting
- You don't need to understand this in order to use virtualenv
- A crucial part of developing Python is a big, clever hack
- Hacks are brittle
--
Python upgrades or OS could cause problems by changing site.py
???
virtualenv
has to play catch-up- Last release of
virtualenv
on 11/16/2016, 3.6 released 12/23/2016
layout: true
.small[
api_venv
├── bin
*│ ├── activate
│ ├── pip
│ ├── pip3
*│ ├── python -> python3
│ ├── python3 -> /usr/local/bin/python3
│ # more executables and activate scripts
├── include
├── lib
│ └── python3.6
*│ └── site-packages
│ ├── certifi
│ ├── chardet
│ ├── idna
│ ├── pip
│ ├── requests
│ ├── setuptools
│ ├── urllib3
│ # more packaging resources
└── pyvenv.cfg
]
???
- Back to the venv that we created earlier
tree -L 4 --dirsfirst -I '__pycache__|*.dist-info' api_venv
- Much shorter, even with installed packages!
- Still have
site-packages
,activate
, etc., but no symbolic links - Handy
python
symbolic link
venv
added to standard library in Python 3.3
???
- Python 3.3 released on 9/29/2012
--
No need for hacked site.py
and standard library symbolic links
Adds pyvenv.cfg
to tell Python that this is a venv
$ cat api_venv/pyvenv.cfg
home = /usr/local/bin
# ...
???
- And where to find the standard library
layout: false
$ source api_venv/bin/activate
???
source
allows script to modify current shell session- Doesn't work without it
--
(api_venv)$ env | grep -e VIRTUAL_ENV -e PATH
VIRTUAL_ENV=/Users/brian/Code/api_venv
PATH=/Users/brian/Code/api_venv/bin:/Users/brian/.local/bin:/usr/local/bin:/usr/bin:...
???
- Modifies
$PS1
using directory name $VIRTUAL_ENV
is root of venv- Modifies
$PATH
- Before
activate
wouldn't haveapi_venv/bin
--
(api_venv)$ which python
/Users/brian/Code/api_venv/bin/python
--
Just a shortcut to python
--
Use the virtualenv without activate
via api_venv/bin/python
--
(api_venv)$ deactivate
$ env | grep -e VIRTUAL_ENV -e PATH
PATH=/Users/brian/.local/bin:/usr/local/bin:/usr/bin:...
???
- Cleans up
- Pretty much the same in Python 2 and 3
TODO
Virtualenv's bin/activate
is Doing It Wrong
- Shell-specific scripts
- Brittle
$PS1
modification - Doesn't solve setting/unsetting environment variables
- Sub-shell is better
name: where
Depends on the project, and your preferences
Should be isolated from your source code
???
$ python3 -m venv project
$ cd project
$ source bin/activate
(project)$ vim project.py
Need bin
, etc. in .gitignore
bin
, etc. could conflict with your project
Hard to delete
Not recommended
$ python3 -m venv ~/venvs/project
$ source ~/venvs/project/bin/activate
(project)$ cd ~/Code/project
(project)$ vim project.py
--
Completely isolated from code
Consistent (i.e., aliasable) activate
???
- Because all venvs are in the sampe place...
- Replace
~/venvs
with whatever you want - No need to
.gitignore
--
This is my preference
$ cd ~/Code/project
$ python3 -m venv venv
$ source venv/bin/activate
(venv)$ vim project.py
--
Common pattern, similar to Node and Ruby
Need venv
in .gitignore
Use --prompt
option for (project)$
???
- Isolate the venv in a sub-directory in your project
- Don't put venv in source control
layout: true
name: tips
From my experience...
Delete with rm -rf
Avoid editing files in the venv
Recreate instead of moving or upgrading
???
- Don't take them too seriously
- Not in version control, so edits need to be copied to other development machines
- Not relocatable
--
Use pip install -r requirements.txt
, and check out pip-tools
Check out autoenv and direnv for managing shell environment
???
- Makes it easy to recreate virtualenvs
- Auto-activate venv, set/unset environment variables
--
Think twice before using pip
outside of a venv
--
Prefer venv
over virtualenv
for Python 3
--
Don't need venv for scripts that only use standard library
Start scripts with #!/usr/bin/env python3
???
TODO
Outside of a venv, be explicit about which Python you're using:
$ python3.6 -m pip install --user
$ python2.7 -m virtualenv
name: related layout: false
???
- Make venv easier to work with
- Reduce typing, establish conventions
- Leverage isolation
layout: true
Keeps virtualenvs and project directories isolated and organized
???
- Installation is more than just
pip install
--
$ mkproject -p python3 api_project
Running virtualenv with interpreter /usr/local/bin/python3
New python executable in /Users/brian/.virtualenvs/api_project/bin/python3.6
# ... virtualenv output ...
Setting project for api_project to /Users/brian/Code/api_project
(api_project)$ pwd
/Users/brian/Code/api_project
???
- Shell commands replace
python3 -m venv
- Created venv, project directory, activated, and
cd
'd into project
--
We deactivate
, time passes...
--
$ workon api_project
(api_project)$ pwd
/Users/brian/Code/api_project
???
- Now we're back in the virtualenv and our project directory
Convenient access to virtualenvs by name
Adds pre- and post- hooks for activate
and deactivate
???
- Including tab completion
- Set/unset environment variables
--
Adds hacks on top of virtualenv
and its hacks
Slows down shell startup
???
- Has a lazy-loading option
- Some attempts at a
venv
-based alternative, but nothing as prevelant
--
IMHO, better to use venv
/ virtualenv
directly, and set up your own shortcuts
See my .bash_profile
and .bashrc
for an example
???
- I used virtualenvwrapper for a long time, got frustrated by its magic
layout: false
A better virtualenvwrapper
--
$ pew mkproject -p python3 api_pew
Running virtualenv with interpreter /usr/local/bin/python3
New python executable in /Users/brian/.virtualenvs/api_pew/bin/python3.6
# ... virtualenv output ...
Setting project for api_pew to /Users/brian/Code/api_pew
Launching subshell in virtual environment. Type 'exit' or 'Ctrl+D' to return.
???
- Behaves similarly to
virtualenvwrapoper
- New shell session avoids need for shell-specific hacks
- Only need to update environment
--
(api_pew)$ exit
???
- No
deactivate
; our shell environment is gone
--
$ pew workon api_pew
(api_pew)$
???
- Pure Python, install with
pip
- Still uses
virtualenv
, but there's an open issue about usingvenv
Install command line tools into separate virtualenvs
--
$ pipsi install --python python3 pew
New python executable in /Users/brian/.local/venvs/pew/bin/python3.6
Linked script /Users/brian/.local/bin/pew
# ...
???
- Use instead of
pip install
--
$ /Users/brian/.local/venvs/pew/bin/python -m pip list
pew (0.1.26)
pip (9.0.1)
requests (2.18.4)
virtualenv (15.1.0)
# ...
--
$ pew version
0.1.26
???
- Uses
virtualenv
to be compatible with many versions of Python - Shortcut for manual venv and symlink into
~/bin
???
- I haven't used these much, but they look cool
--
Pipenv: "Sacred Marriage of Pipfile, Pip, & Virtualenv"
???
- One command to manage venvs and packages
- Bleeding edge
--
pyenv: "Simple Python version management"
- Install and switch between multiple versions of Python
pyenv virtualenv 3.6.2 api_pyenv
???
- Deep hooks into shell
--
tox: "Automate and standardize testing in Python"
- Creates a virtualenv for combinations of Python and package versions
- Installs and tests your project in each
???
- For example, a project that supports Django 1.8+, Python 2.7-3.6
name: reference
- Python Virtual Environments - a primer
- Don't Make Us Say We Told You So: virtualenv for New Pythonistas
- Reverse-engineering Ian Bicking's brain: inside pip and virtualenv (Slides)
- venv documentation
- PEP 405 -- Python Virtual Environments
- virtualenv documentation
- Installing Python Modules
???
- Watch to the end of video to see the genesis of
venv
in Python 3
TODO
- IDLE
python3 -m idlelib
python2 -c "from idlelib.PyShell import main; main();
- tmux
- Fabric
- jupyter
- Apache
- vim
- jedi
- Used by, but not exclusive to, the Anaconda distribution of Python
- Handles package installation and virtual environments for multiple languages
- Use a
Dockerfile
to build an image for a specific Python version with a specific set of packages - Run scripts, apps, etc. in containers using that image
name: installing
Ubuntu
apt install python3-venv python3.6-venv
- Don't try to change
/usr/bin/python3
, e.g. viaupdate-alternatives
- Use
python3.6 -m venv
instead
???
- Vagrantfile in repo
TODO
- TL;DR