Skip to content

Commit

Permalink
Merge pull request #277 from carpentries-incubator/issue-276
Browse files Browse the repository at this point in the history
Switch to "python3 -m" to install pip
  • Loading branch information
anenadic authored Nov 22, 2023
2 parents c9312a5 + 3449c4e commit 0ec8270
Show file tree
Hide file tree
Showing 10 changed files with 73 additions and 75 deletions.
72 changes: 35 additions & 37 deletions _episodes/12-virtual-environments.md
Original file line number Diff line number Diff line change
Expand Up @@ -17,8 +17,8 @@ keypoints:
- "Use `pip` to install and manage Python external (third-party) libraries."
- "`pip` allows you to declare all dependencies for a project in a separate
file (by convention called `requirements.txt`) which can be shared with collaborators/users and used to replicate a virtual environment."
- "Use `pip3 freeze > requirements.txt` to take snapshot of your project's dependencies."
- "Use `pip3 install -r requirements.txt` to replicate someone else's virtual environment on your machine from
- "Use `python3 -m pip freeze > requirements.txt` to take snapshot of your project's dependencies."
- "Use `python3 -m pip install -r requirements.txt` to replicate someone else's virtual environment on your machine from
the `requirements.txt` file."
---

Expand Down Expand Up @@ -220,16 +220,19 @@ conventionally within your software project so they are co-located.
This will create the target directory for the virtual environment
(and any parent directories that don’t exist already).
> ## What is `-m` Flag in `python3` Command??
> ## What is `-m` Flag in `python3` Command?
> The Python `-m` flag means "module" and tells the Python interpreter to treat what follows `-m`
> as the name of a module and not as a single, executable program with the same name. Some modules
> (such as `venv` or `pip`) have main entry points and the `-m` flag can be used to invoke them
> on the command line via the `python` command. The main difference between running such modules as
> standalone programs (e.g. executing "venv" by running the `venv` command directly)
> versus using `python3 -m` command seems to be that with latter you are in full control of
> which Python module will be invoked (the one that came with your environment's Python interpreter vs.
> some other version you may have on your system). This makes it a more reliable way to set things
> up correctly and avoid issues that could prove difficult to trace and debug.
> as the name of a module and not as a single, executable program with the same name.
> Some modules (such as `venv` or `pip`) have main entry points
> and the `-m` flag can be used to invoke them on the command line via the `python` command.
> The main difference between running such modules as standalone programs
> (e.g. executing "venv" by running the `venv` command directly)
> versus using `python3 -m` command seems to be that
> with latter you are in full control of which Python module will be invoked
> (the one that came with your environment's Python interpreter vs.
> some other version you may have on your system).
> This makes it a more reliable way to set things up correctly
> and avoid issues that could prove difficult to trace and debug.
{: .callout}
For our project let's create a virtual environment called "venv".
Expand Down Expand Up @@ -378,50 +381,45 @@ To install the latest version of a package with `pip`
you use pip's `install` command and specify the package’s name, e.g.:
~~~
(venv) $ pip3 install numpy
(venv) $ pip3 install matplotlib
(venv) $ python3 -m pip install numpy
(venv) $ python3 -m pip install matplotlib
~~~
{: .language-bash}
or like this to install multiple packages at once for short:
~~~
(venv) $ pip3 install numpy matplotlib
(venv) $ python3 -m pip install numpy matplotlib
~~~
{: .language-bash}
> ## How About `python3 -m pip install`?
> Why are we not using `pip` as an argument to `python3` command,
> in the same way we did with `venv`
> (i.e. `python3 -m venv`)?
> `python3 -m pip install` should be used according to the
> [official Pip documentation](https://pip.pypa.io/en/stable/user_guide/#running-pip);
> other official documentation still seems to have a mixture of usages.
> Core Python developer Brett Cannon offers a
> ## How About `pip3 install <package-name>` Command?
> You may have seen or used the `pip3 install <package-name>` command in the past, which is shorter
> and perhaps more intuitive than `python3 -m pip install`. However, the
> [official Pip documentation](https://pip.pypa.io/en/stable/user_guide/#running-pip) recommends
> `python3 -m pip install` and core Python developer Brett Cannon offers a
> [more detailed explanation](https://snarky.ca/why-you-should-use-python-m-pip/)
> of edge cases when the two options may produce different results
> and recommends `python3 -m pip install`.
> We kept the old-style command (`pip3 install`)
> as it seems more prevalent among developers at the moment -
> but it may be a convention that will soon change and certainly something you should consider.
> of edge cases when the two commands may produce different results and why `python3 -m pip install`
> is recommended. In this material, we will use `python3 -m` whenever we have to invoke a Python
> module from command line.
{: .callout}
If you run the `pip3 install` command on a package that is already installed,
If you run the `python3 -m pip install` command on a package that is already installed,
`pip` will notice this and do nothing.
To install a specific version of a Python package
give the package name followed by `==` and the version number,
e.g. `pip3 install numpy==1.21.1`.
e.g. `python3 -m pip install numpy==1.21.1`.
To specify a minimum version of a Python package,
you can do `pip3 install numpy>=1.20`.
you can do `python3 -m pip install numpy>=1.20`.
To upgrade a package to the latest version, e.g. `pip3 install --upgrade numpy`.
To upgrade a package to the latest version, e.g. `python3 -m pip install --upgrade numpy`.
To display information about a particular installed package do:
~~~
(venv) $ pip3 show numpy
(venv) $ python3 -m pip show numpy
~~~
{: .language-bash}
~~~
Expand All @@ -441,7 +439,7 @@ Required-by: contourpy, matplotlib
To list all packages installed with `pip` (in your current virtual environment):
~~~
(venv) $ pip3 list
(venv) $ python3 -m pip list
~~~
{: .language-bash}
~~~
Expand All @@ -463,7 +461,7 @@ six 1.16.0
~~~
{: .output}
To uninstall a package installed in the virtual environment do: `pip3 uninstall package-name`.
To uninstall a package installed in the virtual environment do: `python3 -m pip uninstall <package-name>`.
You can also supply a list of packages to uninstall at the same time.
### Exporting/Importing Virtual Environments Using `pip`
Expand All @@ -475,11 +473,11 @@ and everyone can replicate equivalent virtual environments on their machines.
`pip` has a handy way of exporting, saving and sharing virtual environments.
To export your active environment -
use `pip3 freeze` command to produce a list of packages installed in the virtual environment.
use `python3 -m pip freeze` command to produce a list of packages installed in the virtual environment.
A common convention is to put this list in a `requirements.txt` file:
~~~
(venv) $ pip3 freeze > requirements.txt
(venv) $ python3 -m pip freeze > requirements.txt
(venv) $ cat requirements.txt
~~~
{: .language-bash}
Expand Down Expand Up @@ -510,7 +508,7 @@ They can then replicate your environment
and install all the necessary packages from the project root as follows:
~~~
(venv) $ pip3 install -r requirements.txt
(venv) $ python3 -m pip install -r requirements.txt
~~~
{: .language-bash}
Expand Down
21 changes: 11 additions & 10 deletions _episodes/13-ides.md
Original file line number Diff line number Diff line change
Expand Up @@ -183,9 +183,9 @@ Let's see this in action through the following exercise.
>> ## Solution
>> From the previous episode,
>> you may remember that we can get the list of packages in the current virtual environment
>> using the `pip3 list` command:
>> using `pip`:
>> ~~~
>> (venv) $ pip3 list
>> (venv) $ python3 -m pip list
>> ~~~
>> {: .language-bash}
>> ~~~
Expand All @@ -206,11 +206,11 @@ setuptools 67.6.1
six 1.16.0
>> ~~~
>> {: .output}
>> However, `pip3 list` shows all the packages in the virtual environment -
>> However, `python3 -m pip list` shows all the packages in the virtual environment -
>> if we want to see only the list of packages that we installed,
>> we can use the `pip3 freeze` command instead:
>> we can use the `python3 -m pip freeze` command instead:
>> ~~~
>> (venv) $ pip3 freeze
>> (venv) $ python3 -m pip freeze
>> ~~~
>> {: .language-bash}
>> ~~~
Expand All @@ -227,13 +227,14 @@ python-dateutil==2.8.2
six==1.16.0
>> ~~~
>> {: .output}
>> We see `pip` in `pip3 list` but not in `pip3 freeze` as we did not install it using `pip`.
>> Remember that we use `pip3 freeze` to update our `requirements.txt` file,
>> We see the `pip` package in `python3 -m pip list` but not in `python3 -m pip freeze`
>> as we did not install it using `pip`.
>> Remember that we use `python3 -m pip freeze` to update our `requirements.txt` file,
>> to keep a list of the packages our virtual environment includes.
>> Python will not do this automatically;
>> we have to manually update the file when our requirements change using:
>> ~~~
>> pip3 freeze > requirements.txt
>> python3 -m pip freeze > requirements.txt
>> ~~~
>> {: .language-bash}
>>
Expand Down Expand Up @@ -320,7 +321,7 @@ Let's do this as an exercise.
>> Let's verify first that the newly installed library `pytest` is appearing in our virtual environment
>> but not in `requirements.txt`. First, let's check the list of installed packages:
>> ~~~
>> (venv) $ pip3 list
>> (venv) $ python3 -m pip list
>> ~~~
>> {: .language-bash}
>> ~~~
Expand Down Expand Up @@ -365,7 +366,7 @@ six==1.16.0
>> {: .output}
>> `pytest` is missing from `requirements.txt`. To add it, we need to update the file by repeating the command:
>> ~~~
>> (venv) $ pip3 freeze > requirements.txt
>> (venv) $ python3 -m pip freeze > requirements.txt
>> ~~~
>> {: .language-bash}
>> `pytest` is now present in `requirements.txt`:
Expand Down
4 changes: 2 additions & 2 deletions _episodes/16-verifying-code-style-linters.md
Original file line number Diff line number Diff line change
Expand Up @@ -34,14 +34,14 @@ $ git switch style-fixes
Pylint is just a Python package so we can install it in our virtual environment using:

~~~
$ pip3 install pylint
$ python3 -m pip install pylint
~~~
{: .language-bash}

We should also update our `requirements.txt` with this new addition:

~~~
$ pip3 freeze > requirements.txt
$ python3 -m pip freeze > requirements.txt
~~~
{: .language-bash}

Expand Down
31 changes: 15 additions & 16 deletions _episodes/21-automatically-testing-software.md
Original file line number Diff line number Diff line change
Expand Up @@ -456,7 +456,7 @@ exit the Python console first (either with `Ctrl-D` or by typing `exit()`),
then do:

~~~
$ pip3 install pytest
$ python3 -m pip install pytest
~~~
{: .language-bash}

Expand All @@ -470,26 +470,25 @@ our virtual environment will now have the `pytest` package installed for use.
Now we can run these tests using `pytest`:

~~~
$ python -m pytest tests/test_models.py
$ python3 -m pytest tests/test_models.py
~~~
{: .language-bash}

Here, we use `-m` to invoke the `pytest` module,
Here, we use `-m` flag of the `python3` command to invoke the `pytest` module,
and specify the `tests/test_models.py` file to run the tests in that file explicitly.

> ## Why Run Pytest Using `python -m` and Not `pytest` ?
> ## Why Run Pytest Using `python3 -m pytest` and Not `pytest`?
>
> Another way to run `pytest` is via its own command,
> so we *could* try to use `pytest tests/test_models.py` on the command line instead,
> but this would lead to a `ModuleNotFoundError: No module named 'inflammation'`.
> This is because using the `python -m pytest` method
> adds the current directory to its list of directories to search for modules,
> whilst using `pytest` does not -
> the `inflammation` subdirectory's contents are not 'seen',
> hence the `ModuleNotFoundError`.
> There are ways to get around this with
> [various methods](https://stackoverflow.com/questions/71297697/modulenotfounderror-when-running-a-simple-pytest),
> but we've used `python -m` for simplicity.
> `pytest` is another Python module that can be run via its own command but this is a good example
> why invoking Python modules via `python3 -m` may be better (recall the [explanation of Python interpreter's `-m` flag](../12-virtual-environments/index.html#what-is--m-flag-in-python3-command)).
> Had we used `pytest tests/test_models.py` command directly,
> this would have led to a "ModuleNotFoundError: No module named 'inflammation'" error. This is
> because `pytest` command (unlike `python3 -m pytest`) does not add the current directory to its list of
> directories to search for modules, hence the `inflammation` subdirectory's contents are not being
> 'seen' by `pytest` causing the `ModuleNotFoundError`. There are ways to work around this problem
> but `python3 -m pytest` ensures it does not happen in the first place.
>
>
{: .callout}

~~~
Expand Down Expand Up @@ -603,7 +602,7 @@ Since we've installed `pytest` to our environment,
we should also regenerate our `requirements.txt`:
~~~
$ pip3 freeze > requirements.txt
$ python3 -m pip freeze > requirements.txt
~~~
{: .language-bash}
Expand Down
4 changes: 2 additions & 2 deletions _episodes/22-scaling-up-unit-testing.md
Original file line number Diff line number Diff line change
Expand Up @@ -147,7 +147,7 @@ to install an additional package `pytest-cov` to our virtual environment,
which is used by `pytest` to tell us how many statements in our code are being tested.
~~~
$ pip3 install pytest-cov
$ python3 -m pip install pytest-cov
$ python3 -m pytest --cov=inflammation.models tests/test_models.py
~~~
{: .language-bash}
Expand Down Expand Up @@ -211,7 +211,7 @@ Again, we should also update our `requirements.txt` file with our latest package
which now also includes `pytest-cov`, and commit it:
~~~
$ pip3 freeze > requirements.txt
$ python3 -m pip freeze > requirements.txt
$ cat requirements.txt
~~~
{: .language-bash}
Expand Down
4 changes: 2 additions & 2 deletions _episodes/23-continuous-integration-automated-testing.md
Original file line number Diff line number Diff line change
Expand Up @@ -208,7 +208,7 @@ jobs:
- name: Install Python dependencies
run: |
python3 -m pip install --upgrade pip
pip3 install -r requirements.txt
python3 -m pip install -r requirements.txt
- name: Test with PyTest
run: |
Expand Down Expand Up @@ -253,7 +253,7 @@ Each of these steps are:
In order to locally install our `inflammation` package
it's good practice to upgrade the version of pip that is present first,
then we use pip to install our package dependencies.
Once installed, we can use `pip3 install -e .` as before to install our own package.
Once installed, we can use `python3 -m pip install -e .` as before to install our own package.
We use `run` here to run theses commands in the CI shell environment
- **Test with PyTest:** lastly, we run `python3 -m pytest`,
with the same arguments we used manually before
Expand Down
4 changes: 2 additions & 2 deletions _episodes/43-software-release.md
Original file line number Diff line number Diff line change
Expand Up @@ -57,7 +57,7 @@ We can install Poetry much like any other Python distributable package, using `p

~~~
$ source venv/bin/activate
$ pip3 install poetry
$ python3 -m pip install poetry
~~~
{: .language-bash}

Expand Down Expand Up @@ -248,7 +248,7 @@ you don't need to run this command yourself,
you've already installed it using `poetry install` above.

~~~
$ pip3 install dist/inflammation*.whl
$ python3 -m pip install dist/inflammation*.whl
~~~
{: .language-bash}

Expand Down
4 changes: 2 additions & 2 deletions _extras/common-issues.md
Original file line number Diff line number Diff line change
Expand Up @@ -173,7 +173,7 @@ In order to get `pip` to use the proxy,
you need to add an additional parameter when installing packages with `pip`:

~~~
$ pip3 install --proxy <proxy-url> <name of package>`
$ python3 -m pip install --proxy <proxy-url> <name of package>`
~~~
{: .language-bash}

Expand Down Expand Up @@ -217,7 +217,7 @@ A workaround is to:
- Close PyCharm
- Downgrade the version of `pip` used by `venv`, e.g. in a command line terminal type:
~~~
$ pip3 install pip==20.2.4
$ python3 -m pip install pip==20.2.4
~~~
{: .language-bash}
- Restart PyCharm
Expand Down
2 changes: 1 addition & 1 deletion _extras/databases.md
Original file line number Diff line number Diff line change
Expand Up @@ -75,7 +75,7 @@ which contains an **Object Relational Mapping** (ORM) framework.
Our first step is to install SQLAlchemy, then we can create our first **mapping**.

```
$ pip3 install sqlalchemy
$ python3 -m pip install sqlalchemy
```
{: .language-bash}

Expand Down
2 changes: 1 addition & 1 deletion _extras/vscode.md
Original file line number Diff line number Diff line change
Expand Up @@ -147,7 +147,7 @@ since then you will be able to run them from the terminal as well.
For example, if you want `pylint` and `black` packages, execute the following from the terminal:

~~~bash
$ pip3 install pylint black
$ python3 -m pip install pylint black
~~~

They will now both be available to run as command line applications,
Expand Down

0 comments on commit 0ec8270

Please sign in to comment.