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'
