07b: VSCode Tutorial

In this tutorial we will explore using an integrated development environment (IDE) for python programming. IDEs offer many advantages over Jupyter Notebooks, particularly for developing production code that will be reused many times and shared with others. The advantages of IDEs include:

  • Linting to catch errors, dead code, and other code quality issues

  • automatic docstring generation and other auto completion

  • code navigation (jumping to function definitions, etc.)

  • interactive debugging (including plotting)

  • support for automated testing and version control

  • unlike Notebooks, code is always executed in the same order (as it would be by another user), thereby increasing the chances for reproducibility.

VSCode offers the above advantages, and many other features through Extensions (plugins) that can be created by anyone. VSCode is also free and works well with large files (GB in size). The Live Share plugin allows for real-time code collaboration and debugging with multiple people.

Getting started

If you haven’t yet, install VS Code here.

Converting the Theis exercise notebook to a Python script

Once VSCode is installed,

  • Open notebooks/part0_python_intro/solutions/07a_Theis-exercise-solution.ipynb in Jupyter

  • From the File menu, select Save and Export Notebook as and then Executable Script. This will create a python script version of the notebook (07a_Theis-exercise-solution.py)

  • Save 07a_Theis-exercise-solution.py to the notebooks/part0_python_intro folder (or copy it there from your Downloads folder).

Launching VSCode

  • Open a fresh Miniforge prompt from the Start menu and navigate to the root folder for the class (containing the AGENDA.md file).

  • Activate the class python environment (conda activate pyclass) if needed.

  • Then type code .

  • If prompted, click “Yes, I trust the authors”

Getting started

See here for an overview of the VSCode user interface.

Once VSCode is launched, click on the Extensions icon on the activity bar on the left. Install the following extensions:

  • Python

  • Python indent

  • Pylance

  • Partial Diff

  • Jupyter

  • Code spell checker

  • autoDocstring

You may find other cool extensions that you want too. A key indicator of an extension’s quality is the number of downloads.

Linting

Now let’s open the script we copied earlier (notebooks/part0_python_intro/07a_Theis-exercise-solution.py). You can do this from the File Explorer on the left side of the screen.

  • as you scroll down through the script, note that “ts” is grayed out in the statement for ts in t: on line 213. This is the linter telling us that this variable is declared but never used. This is “dead code” that we want remove by refactoring.

  • similarly, note further down that the variables x and y are underlined on lines 253 and 254. The linter is warning us that these variables were never declared. This is because their declarations were wrapped in the get_ipython().run_cell_magic statement on line 232, in translation of the notebook to a script.

Cleaning up the script

Let’s clean up the script so that

To make the script easier to work with, you may want to also turn on Word Wrap (from the View menu) and fold all of the code (control + k + 0).

Using the Partial Diff plugin to compare two pieces of code

Let’s use Partial Diff to compare our script to the class solution. Assuming the Partial Diff Extension was installed earlier,

  • open the class solution (notebooks/part0_python_intro/solutions/07a_Theis-exercise-solution.py)

  • within the solution, highlight any body of code; then right click and Select Text for Compare

  • then within your script, highlight the body of code to compare; right click and Compare Text with Previous Solution.

  • Partial Diff will open up a third window showing the differences

Debugging

  • Place a break point anywhere below the first theis() call in the __main__ part of the script.

  • Then go to either Run --> Start Debugging or click on the debug icon in the Activity Bar and choose Run and Debug. Choose Python File if prompted for a configuration. The debugger should run to the break point.

Often it is prudent to include internal checks in code, regardless of the context. There are a number of ways to do this; a simple one is an assert statement that checks a condition.

  • Add an s = in front of the first theis() call to assign the results to a variable, and then right below, the following assert statement:

    assert np.allclose(s, 1.40636669)
    

    This checks that the resulting drawdown is equal to the number on the right within a certain tolerance.

  • try pasting the above assert statement into the debug console at the bottom of your screen. You can also highlight it, right click and Evaluate in Debug Console.

The debug working directory

Import pathlib and type pathlib.Path.cwd() in the debug console. Note that the current directory is the root folder for the class (where we launched VSCode). VSCode is structured around projects, which include everything in a folder that was opened (and any subfolders). By default, the working directory for debugging is set at the root level for the project. We can change this (and other debugging settings), by creating a configuration file called launch.json, which lives inside of a .vscode/ folder at the root level of the project.

  • After stopping the debugging session, make a default launch.json by clicking on the debug icon in the Activity Bar and then create a launch.json file. Choose Python File if prompted for a configuration. A new tab will open up with launch.json. Add "cwd": "${fileDirname}" at the bottom (don’t forget the preceding comma!) so that the file looks like this:

    {
        // Use IntelliSense to learn about possible attributes.
        // Hover to view descriptions of existing attributes.
        // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387
        "version": "0.2.0",
        "configurations": [
            {
                "name": "Python: Current File",
                "type": "python",
                "request": "launch",
                "program": "${file}",
                "console": "integratedTerminal",
                "cwd": "${fileDirname}"
            }
        ]
    }
    
    
  • run the debugger again. Path.cwd() should show the part0_python_intro folder.

The Run and Debug view

The Run and Debug view on the left (available via the Activity Bar) shows the current variables in the namespace, and an additional “Watch” panel where we can add specific variables that we want to observe as the script executes.

  • Add a watch for the s variable by clicking the + sign on the upper right part of the panel.

  • Now put a breakpoint at the return statement of the theis_xy function.

  • continue with the debugger (via the Run menu or debugging toolbar). Execution should stop again at the return statement where we just placed the breakpoint. Notice that s has changed in the Watch panel. There are also now two layers in the “Call Stack” box below– one for the theis_xy() function that we are in, and one for the enclosing the main script.

  • step out to the main script by clicking the second item in the call stack (labeled <module>). Notice that s has changed again, reflecting the current state of s in that namespace. Notice also that the time-drawdown plot from Step 3 in the original notebook has popped up.

Plotting

During a debug session, as long as matplotlib has been imported, one can make plots via the Debug Console or by putting plotting code in the script.

Automatic docstring generation

Right click at the beginning of the first line below the def statement for any of the functions and choose Generate docstring. The autoDocstring extension should make a new template for a numpy docstring that includes all of the parameters listed in the function signature.

Code navigation

In the main part of the script, right click on one of the function calls (e.g. theis) and choose Go to Definition. VSCode should take you to where the function is defined. Right clicking on theis after the def and then Go to References opens a “peek” window with a list of all of the times theis() is called. Clicking on one of them takes you to that location in the script.

Navigation works across modules and packages too. If you right click on np.meshgrid in the main part of the script and Go to Definition, VSCode takes you to the relevant code in numpy.

Liveshare

We will demo the Live Share extension during the class. In the meantime, you can learn more about it here.