Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How to create a CLI in Python that can be installed with PIP?

As the title suggests, I'm trying to make a python script accessible from the command line. I've found libraries like click and argv that make it easy to access arguments passed from the command line, but the user still has to run the script through Python.

Instead of

python /location/to/myscript.py

I want to be able to just do

myscript

from any directory

From what I understand, I can achieve this on my computer by editing my PATH variables. However, I would like to be able to simply do:

pip install myscript

and then access the script by typing myscript from anywhere. Is there some special code I would put in the setup.py?

like image 966
Param Avatar asked Jun 10 '19 23:06

Param


2 Answers

You can do this with setuptools

an example of a nice setup.py (say your package requires pandas and numpy):

import setuptools
setuptools.setup(
    name='myscript',
    version='1.0',
    scripts=['./scripts/myscript'],
    author='Me',
    description='This runs my script which is great.',
    packages=['lib.myscript'],
    install_requires=[
        'setuptools',
        'pandas >= 0.22.0',
        'numpy >= 1.16.0'
    ],
    python_requires='>=3.5'
)

Your directory should be setup as follows:

[dkennetz package]$ ls
lib scripts setup.py

inside lib would be:

[dkennetz package]$ ls lib
myscript

inside of myscript would be:

[dkennetz package]$ ls lib/myscript
__main__.py
__init__.py
helper_module1.py
helper_module2.py

main would be used to call your function and do whatever you want to do.

inside scripts would be:

[dkennetz package]$ ls scripts
myscript

and the contents of myscript would be:

#!/usr/bin/env bash

if [[ ! $@ ]]; then
    python3 -m myscript -h
else
    python3 -m myscript $@
fi

then to run you do: python setup.py install

which will install your program and all of the dependencies you included in install_requires=[] in your setup.py and install myscript as a command-line module:

[dkennetz ~]$ myscript
like image 153
d_kennetz Avatar answered Oct 12 '22 11:10

d_kennetz


Use console_scripts to hook to a specific Python method (not a whole executable), setup.py file:

from setuptools import setup
setup(
    ...
    entry_points = {
        'console_scripts': ['mybinary=mymodule.command_line:cli'],
    },
    name='mymodule',
    ...
)

the command_line.py script would be:

import mymodule

def cli():
    print("Hello world!")

and the project directory would look like this:

myproject/
    mymodule/
        __init__.py
        command_line.py
        ...
    setup.py

Setuptools will generate a standalone script ‘shim’ which imports your module and calls the registered function.

That shim allows you to call mybinary directly and ensures it's correctly invoked by python. It provides platform-specific shims (i.e., on Windows it generates a .exe).

See packaging documentation for more details.

like image 38
Tombart Avatar answered Oct 12 '22 13:10

Tombart