How to extend distutils with a simple post install script?

Question:

I need to run a simple script after the modules and programs have been installed.
I’m having a little trouble finding straight-forward documentation on how to do this. It looks like I need to inherit from distutils.command.install, override some methods and add this object to the setup script. The specifics are a bit hazy though and it seems like a lot of effort for such a simple hook. Does anyone know an easy way to do this?

Asked By: blokkie

||

Answers:

I dug through distutils source for a day to learn enough about it to make a bunch of custom commands. It’s not pretty, but it does work.

import distutils.core
from distutils.command.install import install
...
class my_install(install):
    def run(self):
        install.run(self)
        # Custom stuff here
        # distutils.command.install actually has some nice helper methods
        # and interfaces. I strongly suggest reading the docstrings.
...
distutils.core.setup(..., cmdclass=dict(install=my_install), ...)
Answered By: user79758

OK, I figured it out. The idea is basically to extend one of the distutils commands and overwrite the run method. To tell distutils to use the new class you can use the cmdclass variable.

from distutils.core import setup
from distutils.command.install_data import install_data

class post_install(install_data):
    def run(self):
        # Call parent 
        install_data.run(self)
        # Execute commands
        print "Running"

setup(name="example",
      cmdclass={"install_data": post_install},
      ...
      )

Hope this will help someone else.

Answered By: blokkie

I couldn’t make Joe Wreschnigs answer work and tweaked his answer analogous to the extending distutils documentation. I came up with this code which works fine on my machine.

from distutils import core
from distutils.command.install import install
...
class my_install(install):
    def run(self):
        install.run(self)
        # Custom stuff here
        # distutils.command.install actually has some nice helper methods
        # and interfaces. I strongly suggest reading the docstrings.
...
distutils.core.setup(..., cmdclass={'install': my_install})

Note: I didn’t edit the answer of Joe since I am uncertain why his answer wasn’t working on my machine.

Answered By: Woltan

I got an error when I tried the accepted answer here (might be because I’m using Python 2.6 in this particular case, not sure). This occurred for both ‘setup.py install’ and ‘pip install’:

sudo python setup.py install

fails with error: error in setup.cfg: command ‘my_install’ has no such option ‘single_version_externally_managed’

AND

sudo pip install . -U

fails more verbosely but also with error: option –single-version-externally-managed not recognized

Variation on the accepted answer

Replacing the imports from distutils with setuptools solved the problem for me:

from setuptools import setup
from setuptools.command.install import install
Answered By: Theuns Alberts
Categories: questions Tags: ,
Answers are sorted by their score. The answer accepted by the question owner as the best is marked with
at the top-right corner.