Skip to main content

A Sampling Profiler for Python

Project description

Build Status Windows Build status

Py-Spy is a sampling profiler for Python programs. It lets you visualize what your Python program is spending time on without restarting the program or modifying the code in any way. Py-Spy is extremely low overhead: it is written in Rust for speed and doesn’t run in the same process as the profiled Python program, nor does it interrupt the running program in any way. This means Py-Spy is safe to use against production Python code.

Py-Spy works on Linux, OSX and Windows, and supports profiling all recent versions of the CPython interpreter (versions 2.3-2.7 and 3.3-3.7).

Installation

Prebuilt binary wheels can be installed from PyPI with:

pip install py-spy

If you’re a Rust user, py-spy can also be installed with:

cargo install py-spy

Usage

py-spy works from the command line and takes either the PID of the program you want to sample from or the command line of the python program you want to run:

py-spy --pid 12345
# OR
py-spy -- python myprogram.py

The default visualization is a top-like live view of your python program:

console viewer demo

console viewer demo

There is also support for generating flame graphs from the running process:

py-spy --flame profile.svg --pid 12345
# OR
py-spy --flame profile.svg -- python myprogram.py

Which will generate a SVG file looking like:

flame graph

flame graph

It also possible to dump out the current call stack for each thread by passing --dump to the command line.

Frequently Asked Questions

Why do we need another Python profiler?

This project aims to let you profile and debug any running Python program, even if the program is serving production traffic.

While there are many other python profiling projects, almost all of them require modifying the profiled program in some way. Usually, the profiling code runs inside of the target python process, which will slow down and change how the program operates. This means it’s not generally safe to use these profilers for debugging issues in production services since they will usually have a noticeable impact on performance. The only other Python profiler that runs totally in a separate process is pyflame, which profiles remote python processes by using the ptrace system call. While pyflame is a great project, it doesn’t support Python 3.7 yet and doesn’t work on OSX or Windows.

How does py-spy work?

Py-spy works by directly reading the memory of the python program using the process_vm_readv system call on Linux, the vm_read call on OSX or the ReadProcessMemory call on Windows.

Figuring out the call stack of the Python program is done by looking at the global PyInterpreterState variable to get all the Python threads running in the interpreter, and then iterating over each PyFrameObject in each thread to get the call stack. Since the Python ABI changes between versions, we use rust’s bindgen to generate different rust structures for each Python interpreter class we care about and use these generated structs to figure out the memory layout in the Python program.

Getting the memory address of the Python Interpreter can be a little tricky due to Address Space Layout Randomization. If the target python interpreter ships with symbols it is pretty easy to figure out the memory address of the interpreter by dereferencing the interp_head or _PyRuntime variables depending on the Python version. However, many Python versions are shipped with either stripped binaries or shipped without the corresponding PDB symbol files on Windows. In these cases we scan through the BSS section for addresses that look like they may point to a valid PyInterpreterState and check if the layout of that address is what we expect.

Can py-spy profile native extensions?

Since we’re getting the call stacks of the python program by looking at the PyInterpreterState we don’t yet get information about non-python threads and can’t profile native extensions like those written in languages like Cython or C++. Native code will instead show up as spending time in the line of Python that calls the native function, rather than as its own entry right now.

It should be possible to use something like libunwind to profile the native code in the Python Extensions. If this is something that interests you please upvote this issue.

When do you need to run as sudo?

Py-spy works by reading memory from a different python process, and this might not be allowed for security reasons depending on your OS and system settings. In many cases, running as a root user (with sudo or similar) gets around these security restrictions. OSX always requires running as root, but on Linux it depends on how you are launching py-spy and the system security settings.

On Linux the default configuration is to require root permissions when attaching to a process that isn’t a child. For py-spy this means you can profile without root access by getting py-spy to create the process (py-spy -- python myprogram.py) but attaching to an existing process by specifying a PID will usually require root (sudo py-spy -pid 123456). You can remove this restriction on linux by setting the ptrace_scope sysctl variable.

Why am I having issues profiling /usr/bin/python on OSX?

OSX has a feature called System Integrity Protection that prevents even the root user from reading memory from any binary located in /usr/bin. Unfortunately, this includes the python interpreter that ships with OSX.

There are a couple of different ways to deal with this: * You can install a different Python distribution (you probably want to migrate away from python2 anyways =) * You can use virtualenv to run the system python in an environment where SIP doesn’t apply. * You can disable System Integrity Protection.

Running py-spy in Docker

Running py-spy inside of a docker container will also usually bring up a permissions denied error even when running as root. This error is caused by docker restricting the process_vm_readv system call we are using. This can be overridden by setting `--cap-add SYS_PTRACE <https://docs.docker.com/engine/security/seccomp/>`__ when starting the docker container.

Running under Kubernetes

py-spy needs SYS_PTRACE to be able to read process memory. Kubernetes drops that capability by default, resulting in the error

Permission Denied: Try running again with elevated permissions by going 'sudo env "PATH=$PATH" !!'

The recommended way to deal with this is to edit the spec and add that capability. For a deployment, this is done by adding this to Deployment.spec.template.spec.containers

securityContext:
  capabilities:
    add:
    - SYS_PTRACE

More details on this here: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-capabilities-for-a-container Note that this will remove the existing pods and create those again.

How do you avoid pausing the Python program?

Py-spy doesn’t pause the target python program we are sampling from, and reads the interpreter state from the python process as it is running. Since the calls we use to read memory from are not atomic, and we have to issue multiple calls to get a stack trace this means that occasionally we get errors when sampling. If you run the flame graph code it will report the number of errors at the end (and the top view will report the error rate if it goes beyond 1%). This happens infrequently enough that it shouldn’t affect the output that much though. Julia Evans wrote a great blog post talking about this trade off with rbspy

How are you distributing Rust executable binaries over PyPI?

Ok, so no-one has ever actually asked me this - but I wanted to share since it’s a pretty terrible hack that might be useful to other people.

I really wanted to distribute this package over PyPI, since installing with pip will make this much easier for most Python programmers to get installed on their system. Unfortunately, installing executables as python scripts isn’t something that setuptools supports.

To get around this I’m using setuptools_rust package to build the py-spy binary, and then overriding the distutils install command to copy the built binary into the python scripts folder. By doing this with prebuilt wheels for supported platforms means that we can install py-spy with pip, and not require a Rust compiler on the machine that this is being installed onto.

Does this run on BSD? Support 32-bit Windows? Integrate with PyPy? Work with USC-16 versions of Python2?

Not yet =).

Credits

py-spy is heavily inspired by Julia Evans excellent work on rbspy. In particular, the code to generate the flamegraphs is taken directly from rbspy, and this project uses the (read-process-memory and proc-maps) crates that were spun off from rbspy.

License

Py-spy is released under the GNU General Public License v3.0, see LICENSE file for the full text.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distributions

No source distribution files available for this release.See tutorial on generating distribution archives.

Built Distributions

py_spy-0.1.8-py2.py3-none-win_amd64.whl (1.3 MB view details)

Uploaded Python 2 Python 3 Windows x86-64

py_spy-0.1.8-py2.py3-none-manylinux1_x86_64.whl (2.9 MB view details)

Uploaded Python 2 Python 3

py_spy-0.1.8-py2.py3-none-manylinux1_i686.whl (3.0 MB view details)

Uploaded Python 2 Python 3

py_spy-0.1.8-py2.py3-none-macosx_10_7_x86_64.whl (1.6 MB view details)

Uploaded Python 2 Python 3 macOS 10.7+ x86-64

File details

Details for the file py_spy-0.1.8-py2.py3-none-win_amd64.whl.

File metadata

  • Download URL: py_spy-0.1.8-py2.py3-none-win_amd64.whl
  • Upload date:
  • Size: 1.3 MB
  • Tags: Python 2, Python 3, Windows x86-64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/1.11.0 pkginfo/1.4.2 requests/2.19.1 setuptools/39.0.1 requests-toolbelt/0.8.0 tqdm/4.26.0 CPython/2.7.15

File hashes

Hashes for py_spy-0.1.8-py2.py3-none-win_amd64.whl
Algorithm Hash digest
SHA256 b26ed14746ebec0dd6ca6b1e0af1db182de63a510bc4ab0f1e858eb01ecdd331
MD5 dfcbfcbed11df80907d7d3014deff78c
BLAKE2b-256 884e2eac7ded45799c1ae56d30b7321cfecece348176d249b808c1f636b9c59f

See more details on using hashes here.

File details

Details for the file py_spy-0.1.8-py2.py3-none-manylinux1_x86_64.whl.

File metadata

  • Download URL: py_spy-0.1.8-py2.py3-none-manylinux1_x86_64.whl
  • Upload date:
  • Size: 2.9 MB
  • Tags: Python 2, Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/1.11.0 pkginfo/1.4.2 requests/2.18.4 setuptools/36.5.0.post20170921 requests-toolbelt/0.8.0 tqdm/4.23.4 CPython/3.6.3

File hashes

Hashes for py_spy-0.1.8-py2.py3-none-manylinux1_x86_64.whl
Algorithm Hash digest
SHA256 a98a0c1d916a5b415a35d2fb6fe7a5b5808dd7879d55793d025f1e15ae04693a
MD5 baf130ebc110ed58b8f9ce31ab94bedb
BLAKE2b-256 f8da55333d9cc08227b6fda033d0f7b8de3d8551c5ba20f2c636f55f6bc290ba

See more details on using hashes here.

File details

Details for the file py_spy-0.1.8-py2.py3-none-manylinux1_i686.whl.

File metadata

  • Download URL: py_spy-0.1.8-py2.py3-none-manylinux1_i686.whl
  • Upload date:
  • Size: 3.0 MB
  • Tags: Python 2, Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/1.11.0 pkginfo/1.4.2 requests/2.18.4 setuptools/36.5.0.post20170921 requests-toolbelt/0.8.0 tqdm/4.23.4 CPython/3.6.3

File hashes

Hashes for py_spy-0.1.8-py2.py3-none-manylinux1_i686.whl
Algorithm Hash digest
SHA256 6c88435d7411acb148dee283ef2cc39bff42a178d52725fba3fe89205a02fa90
MD5 71ff62b62c3a9733bc524b2de3fb7c8d
BLAKE2b-256 920212a5fb6002052ba4c7ee93307aa756e2c87077a00f620223eeecabdb5245

See more details on using hashes here.

File details

Details for the file py_spy-0.1.8-py2.py3-none-macosx_10_7_x86_64.whl.

File metadata

  • Download URL: py_spy-0.1.8-py2.py3-none-macosx_10_7_x86_64.whl
  • Upload date:
  • Size: 1.6 MB
  • Tags: Python 2, Python 3, macOS 10.7+ x86-64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/1.11.0 pkginfo/1.4.2 requests/2.18.4 setuptools/36.5.0.post20170921 requests-toolbelt/0.8.0 tqdm/4.23.4 CPython/3.6.3

File hashes

Hashes for py_spy-0.1.8-py2.py3-none-macosx_10_7_x86_64.whl
Algorithm Hash digest
SHA256 5b26878961cc1b5d1f18e21f8ed6817aced5f4ae835d5f7fa277982e5c3d9daf
MD5 9300000e8db831508e18ff0c4c4dc694
BLAKE2b-256 4d5dea5eed92c564777e5e142e37569f44eba4e91032345a19b0ed2ed891bd0c

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page