7. Exemplos

Este capítulo fornece vários exemplos básicos para ajudar a começar com distutils. Informações adicionais sobre o uso de distutils podem ser encontradas no Distutils Cookbook.

Ver também

Distutils Cookbook

Coleção de receitas mostrando como obter mais controle sobre distutils.

7.1. Distribuição Python pura (por módulo)

Se você está apenas distribuindo alguns módulos, especialmente se eles não residem em um pacote específico, você pode especificá-los individualmente usando a opção py_modules no script de instalação.

No caso mais simples, você terá dois arquivos com os quais se preocupar: um script de instalação e o único módulo que você está distribuindo foo.py neste exemplo:

<root>/
        setup.py
        foo.py

(Em todos os diagramas desta seção, <root> se refere ao diretório raiz da distribuição.) Um script de instalação mínimo para descrever essa situação seria:

from distutils.core import setup
setup(name='foo',
      version='1.0',
      py_modules=['foo'],
      )

Observe que o nome da distribuição é especificado independentemente com a opção name, e não existe uma regra que diga que deve ser igual ao nome do único módulo na distribuição (embora seja provavelmente uma boa convenção a seguir ) No entanto, o nome da distribuição é usado para gerar nomes de arquivos, portanto, você deve usar letras, dígitos, sublinhados e hífenes.

Como py_modules é uma lista, é claro que você pode especificar vários módulos, por exemplo. se você estiver distribuindo os módulos foo e bar, sua configuração poderá ser assim:

<root>/
        setup.py
        foo.py
        bar.py

e o script de configuração pode ser

from distutils.core import setup
setup(name='foobar',
      version='1.0',
      py_modules=['foo', 'bar'],
      )

Você pode colocar os arquivos fonte do módulo em outro diretório, mas se você tiver módulos suficientes para fazer isso, provavelmente será mais fácil especificar módulos por pacote do que listá-los individualmente.

7.2. Distribuição Python pura (por pacote)

Se você tiver mais do que alguns módulos para distribuir, especialmente se estiverem em vários pacotes, provavelmente será mais fácil especificar pacotes inteiros do que módulos individuais. Isso funciona mesmo que seus módulos não estejam em um pacote; você pode simplesmente dizer ao Distutils para processar os módulos do pacote raiz, e isso funciona da mesma forma que em qualquer outro pacote (exceto que você não precisa ter um arquivo __init__.py).

O script de instalação do último exemplo também pode ser escrito como

from distutils.core import setup
setup(name='foobar',
      version='1.0',
      packages=[''],
      )

(A string vazia representa o pacote raiz.)

Se esses dois arquivos forem movidos para um subdiretório, mas permanecerem no pacote raiz, por exemplo:

<root>/
        setup.py
        src/      foo.py
                  bar.py

você ainda especificaria o pacote raiz, mas precisará informar ao Distutils onde estão os arquivos fonte no pacote raiz:

from distutils.core import setup
setup(name='foobar',
      version='1.0',
      package_dir={'': 'src'},
      packages=[''],
      )

Mais tipicamente, porém, você deseja distribuir vários módulos no mesmo pacote (ou em subpacotes). Por exemplo, se os módulos foo e bar pertencem ao pacote foobar, uma maneira de fazer o layout da sua árvore de fontes é

<root>/
        setup.py
        foobar/
                 __init__.py
                 foo.py
                 bar.py

Na verdade, esse é o layout padrão esperado pelo Distutils e o que exige menos trabalho para descrever no seu script de instalação:

from distutils.core import setup
setup(name='foobar',
      version='1.0',
      packages=['foobar'],
      )

Se você quiser colocar módulos em diretórios não nomeados para o pacote, precisará usar a opção package_dir novamente. Por exemplo, se o diretório src contiver módulos no pacote foobar:

<root>/
        setup.py
        src/
                 __init__.py
                 foo.py
                 bar.py

um script de instalação apropriado seria

from distutils.core import setup
setup(name='foobar',
      version='1.0',
      package_dir={'foobar': 'src'},
      packages=['foobar'],
      )

Ou, você pode colocar módulos do seu pacote principal diretamente na raiz da distribuição:

<root>/
        setup.py
        __init__.py
        foo.py
        bar.py

nesse caso, seu script de instalação seria

from distutils.core import setup
setup(name='foobar',
      version='1.0',
      package_dir={'foobar': ''},
      packages=['foobar'],
      )

(A string vazia também representa o diretório atual.)

Se você possui subpacotes, eles devem ser listados explicitamente em packages, mas qualquer entrada em package_dir se estende automaticamente aos subpacotes. (Em outras palavras, o Distutils não varre sua árvore de fontes, tentando descobrir quais diretórios correspondem aos pacotes Python procurando __init__.py.) Portanto, se o layout padrão aumentar um subpacote:

<root>/
        setup.py
        foobar/
                 __init__.py
                 foo.py
                 bar.py
                 subfoo/
                           __init__.py
                           blah.py

então o script de instalação correspondente seria

from distutils.core import setup
setup(name='foobar',
      version='1.0',
      packages=['foobar', 'foobar.subfoo'],
      )

7.3. Módulo de extensão única

Os módulos de extensão são especificados usando a opção ext_modules. package_dir não afeta onde os arquivos de origem das extensões são encontrados; isso afeta apenas a fonte dos módulos Python puros. O caso mais simples, um único módulo de extensão em um único arquivo de origem C, é:

<root>/
        setup.py
        foo.c

Se a extensão foo pertencer ao pacote raiz, o script de instalação para isso pode ser

from distutils.core import setup
from distutils.extension import Extension
setup(name='foobar',
      version='1.0',
      ext_modules=[Extension('foo', ['foo.c'])],
      )

Se a extensão realmente pertence a um pacote, diga foopkg, então

Com exatamente o mesmo layout da árvore de fontes, esta extensão pode ser colocada no pacote foopkg simplesmente alterando o nome da extensão:

from distutils.core import setup
from distutils.extension import Extension
setup(name='foobar',
      version='1.0',
      ext_modules=[Extension('foopkg.foo', ['foo.c'])],
      )

7.4. Verificando um Pacote

O comando check permite verificar se os metadados do seu pacote atendem aos requisitos mínimos para compilar uma distribuição.

Para executá-lo, basta chamá-lo usando o script setup.py. Se algo estiver faltando, check exibirá um aviso.

Vamos dar um exemplo com um script simples:

from distutils.core import setup

setup(name='foobar')

A execução do comando check exibirá alguns avisos:

$ python setup.py check
running check
warning: check: missing required meta-data: version, url
warning: check: missing meta-data: either (author and author_email) or
         (maintainer and maintainer_email) must be supplied

Se você usar a sintaxe reStructuredText no campo long_description e o docutils estiver instalado, poderá verificar se a sintaxe está correta com o comando check, usando a opção restructuredtext.

Por exemplo, se o script setup.py for alterado assim:

from distutils.core import setup

desc = """\
My description
==============

This is the description of the ``foobar`` package.
"""

setup(name='foobar', version='1', author='tarek',
    author_email='tarek@ziade.org',
    url='http://example.com', long_description=desc)

Onde a descrição longa está quebrada, check poderá detectá-la usando o analisador docutils:

$ python setup.py check --restructuredtext
running check
warning: check: Title underline too short. (line 2)
warning: check: Could not finish the parsing.

7.5. Lendo os metadados

A função distutils.core.setup() fornece uma interface de linha de comando que permite consultar os campos de metadados de um projeto através do script setup.py de um projeto fornecido:

$ python setup.py --name
distribute

Essa chamada lê os metadados name executando a função distutils.core.setup(). Embora, quando uma distribuição de origem ou binária é criada com o Distutils, os campos de metadados sejam gravados em um arquivo estático chamado PKG-INFO. Quando um projeto baseado no Distutils é instalado no Python, o arquivo PKG-INFO é copiado juntamente com os módulos e pacotes da distribuição em NOME-VERSÃO-pyX.X.egg-info, em que NOME é o nome do projeto, VERSÃO sua versão conforme definida nos metadados e pyX.X a versão principal e secundária do Python, como 2.7 ou 3.2.

Você pode ler novamente esse arquivo estático usando a classe distutils.dist.DistributionMetadata e seu método read_pkg_file():

>>> from distutils.dist import DistributionMetadata
>>> metadata = DistributionMetadata()
>>> metadata.read_pkg_file(open('distribute-0.6.8-py2.7.egg-info'))
>>> metadata.name
'distribute'
>>> metadata.version
'0.6.8'
>>> metadata.description
'Easily download, build, install, upgrade, and uninstall Python packages'

Observe que a classe também pode ser instanciada com um caminho de arquivo de metadados para carregar seus valores:

>>> pkg_info_path = 'distribute-0.6.8-py2.7.egg-info'
>>> DistributionMetadata(pkg_info_path).name
'distribute'