Hello,

This is the second and hopefully last version of python-r1 +
distutils-r1 eclasses before committing. I would also like to shortly
point out the goals and the differences between various python eclasses
in Gentoo.

Both are attached. For those who prefer hosted form:
https://bitbucket.org/mgorny/gx86-working-tree/src/5a2d39e69d6d/gx86/eclass/python-r1.eclass
https://bitbucket.org/mgorny/gx86-working-tree/src/5a2d39e69d6d/gx86/eclass/distutils-r1.eclass

Changes from the previous version:
- support for DOCS and HTML_DOCS (alike EAPI 4);
- minimal support for passing arguments to setup.py (through
  distutils-r1_python_compile & distutils-r1_python_install);
- EPYTHON values adjusted to match python.eclass;
- scripts are -${EPYTHON} suffixed -- i.e. foo-python2.7,
  foo-pypy-c1.8;
- a wrapper script[1] is used to choose the correct script instead of
  constant symlink. Thus, EPYTHON & eselect-python are respected,
- PYTHON_COMPAT can be an array only.

What still needs considering:
- a nicer way of passing setup.py arguments (mysetuppyargs=()?,
  esetuppy wrapper?);
- Prefix support.

[1]:https://bitbucket.org/mgorny/python-exec


Now, for the differences part.

Goals
-----

python.eclass aims to support every single Python package out there,
including rare corner cases supported in-eclass. It supports both
packages supporting one and multiple Python implementations. distutils
support is provided by second eclass, and both provide phase functions
(python.eclass providing default ph.f. wrappers).

python-distutils-ng aims to support majority of Python packages using
distutils. With a bit of hackery it can support non-distutils packages,
but the use is limited to the common cases. In an uncommon case, it's
not flexible enough. It supports Python packages supporting multiple
implementations only.

python-r1 aims mostly to provide tools to support majority of Python
packages. It tries to be simple & flexible, thus allowing handling
various corner cases without special branches of code in the eclass.
It doesn't export any phase functions, nor set dependencies by default
(just provides a variable with them).

Right now, it supports packages supporting multiple implementations
only; if necessary, the support will be extended -- either through
additional code if that wouldn't add too much complexity, or through
additional eclass.

It is accompanied by distutils-r1 which provides a simple overlay for
the very common case of distutils packages. This eclass exports phase
functions and sets dependencies implicitly. It also handles renaming
the distutils-installed scripts and linking the python-exec wrapper.


Python targets
--------------

python.eclass uses implicit Python target specifications. It provides
no ebuild-transparent way of enabling/disabling them.

python-distutils-ng and python-r1 use USE flags to explicitly list
enabled Python implementations. Both eclasses use the same values for
PYTHON_TARGETS USE_EXPAND.


Installed scripts
-----------------

python-distutils-ng rework the installed scripts creating copies for
Python implementations and changing the shebang. The created copies
don't differ any other way. The old script name is then symlinked to
the version for default implementation.

python.eclass & python-r1 instead let distutils rework the scripts
and just rename them before merging. A wrapper is used to choose
the correct version, respecting EPYTHON & eselect-python.

python.eclass installs the files to separate installation images
and merges them. python-r1 installs them to the main image directly,
renaming the installed scripts between successive implementations.

python.eclass creates a wrapper script for each package. The script is
written in python. python-r1 instead installs the compiled wrapper
once (in an ebuild), and symlinks it for the packages. The wrapper is
written in much simpler C.

The implementation suffixes for all three eclasses are different:
- python.eclass uses -2.6, -2.7 for Python, -2.7-pypy-1.8,
  -2.7-pypy-1.9 for pypy and -2.5-jython for jython (enjoy!);
- p-d-ng uses Python target names (-python2_6, -python2_7, -pypy1_8),
- p-r1 uses $EPYTHON values (-python2.6, -python2.7, -pypy-c1.8 (sic!)).


Dependency on Python
--------------------

python.eclass has a few variables necessary to set dependencies
on Python implementation, including internal sub-syntax (and an even
more complete generator sub-syntax in progress overlay). I suspect that
most of the possible variants can be achieved using it, just please
don't make me try learning it all.

Additionally, packages supporting multiple Python implementations are
required to specify their supported Python versions twice -- with
PYTHON_DEPEND and RESTRICT_PYTHON_ABIS.

USE flag dependencies are specified using three variables, and I'd like
to avoid getting anywhere deeper.

p-d-ng has a very simple dependency setup. It has a variable for
listing supported Python versions and a variable to make Python
dependencies conditional under the 'python' flag. Lately, it has been
added a method of listing requested USE flags.

p-d-ng does not provide any support for more complex dependency
specifications, nor a way to disable adding default Python dependencies.

python-r1 does not append any dependencies by default, and instead
exports them in a variable. The variable can be used to easily express
simple cases, and avoids polluting the ebuild in more complex cases. It
also supports providing USE dependencies for the implementation.

distutils-r1 adds the Python dependencies to RDEPEND & DEPEND;
additionally, it adds a dependency on python-exec.


Dependencies on other packages
------------------------------

As python.eclass (as of gx86 state) does not express enabled
implementations explicitly, it is not possible to request those
implementations from other packages. The dependencies on other Python
packages are thus simple.

p-d-ng does not provide any simpler way of writing dependencies
on other Python packages using the eclass. Thus, the necessary USE
dependencies have to be written by hand.

python-r1 does provide a simple PYTHON_USEDEP variable which can be
used in dependency atoms to request the same Python implementations
being enabled.


Phase functions for distutils
-----------------------------

distutils.eclass allows specifying additional arguments to setup.py.
src_install() installs default documentation files (*unconditionally*)
and ${DOCS}, it also does installation image magic.

p-d-ng has mixed unconditional and user-overridable stuff. Copying
sources is done unconditionally; redoing scripts can be disabled using
a separate variable. setup.py invocations are overridable.

d-r1 splits each phase into 'global' and 'per-implementation' stages
which are both overridable. The default stages apply PATCHES (and user
patches), copy sources, run setup.py, install DOCS & HTML_DOCS and link
python-exec wrappers.



Thank you for reading up to this. I'd appreciate any comments, ideas,
and especially any API suggestions while the eclass still ain't live.

-- 
Best regards,
Michał Górny
# Copyright 1999-2012 Gentoo Foundation
# Distributed under the terms of the GNU General Public License v2
# $Header: $

# @ECLASS: python-r1
# @MAINTAINER:
# Python herd <pyt...@gentoo.org>
# @AUTHOR:
# Author: Michał Górny <mgo...@gentoo.org>
# Based on work of: Krzysztof Pawlik <nelch...@gentoo.org>
# @BLURB: A common, simple eclass for Python packages.
# @DESCRIPTION:
# A common eclass providing helper functions to build and install
# packages supporting being installed for multiple Python
# implementations.
#
# This eclass sets correct IUSE and REQUIRED_USE. It exports PYTHON_DEPS
# and PYTHON_USEDEP so you can create correct dependencies for your
# package easily. It also provides methods to easily run a command for
# each enabled Python implementation and duplicate the sources for them.

case "${EAPI}" in
        0|1|2|3)
                die "Unsupported EAPI=${EAPI} (too old) for ${ECLASS}"
                ;;
        4)
                # EAPI=4 needed for REQUIRED_USE
                ;;
        *)
                die "Unsupported EAPI=${EAPI} (unknown) for ${ECLASS}"
                ;;
esac

# @ECLASS-VARIABLE: _PYTHON_ALL_IMPLS
# @INTERNAL
# @DESCRIPTION:
# All supported Python implementations, most preferred last.
_PYTHON_ALL_IMPLS=(
        jython2_5
        pypy1_8 pypy1_9
        python3_1 python3_2
        python2_5 python2_6 python2_7
)

# @ECLASS-VARIABLE: PYTHON_COMPAT
# @DESCRIPTION:
# This variable contains a list of Python implementations the package
# supports. It must be set before the `inherit' call.  The default is to
# enable all implementations. It has to be an array.
: ${PYTHON_COMPAT:=${_PYTHON_ALL_IMPLS[@]}}

# @ECLASS-VARIABLE: PYTHON_REQ_USE
# @DEFAULT_UNSET
# @DESCRIPTION:
# The list of USEflags required to be enabled on the chosen Python
# implementations, formed as a USE-dependency string. It should be valid
# for all implementations in PYTHON_COMPAT, so it may be necessary to
# use USE defaults.
#
# Example:
# @CODE
# PYTHON_REQ_USE="gdbm,ncurses(-)?"
# @CODE
#
# Will cause the Python dependencies to look like:
# @CODE
# python_targets_pythonX_Y? (
#   dev-lang/python:X_Y[gdbm,ncurses(-)?] )
# @CODE

# @ECLASS-VARIABLE: PYTHON_DEPS
# @DESCRIPTION:
# This is an eclass-generated Python dependency string for all
# implementations listed in PYTHON_COMPAT. It should be used
# in RDEPEND and/or DEPEND like:
#
# @CODE
# RDEPEND="${PYTHON_DEPS}
#   dev-foo/mydep"
# DEPEND="${RDEPEND}"
# @CODE

# @ECLASS-VARIABLE: PYTHON_USEDEP
# @DESCRIPTION:
# This is an eclass-generated USE-dependency string which can be used to
# depend on another Python package being built for the same Python
# implementations. It should be used like:
#
# @CODE
# RDEPEND="dev-python/foo[${PYTHON_USEDEP}]"
# @CODE

_python_set_globals() {
        local flags=( "${PYTHON_COMPAT[@]/#/python_targets_}" )
        local optflags=${flags[@]/%/?}

        IUSE=${flags[*]}
        REQUIRED_USE="|| ( ${flags[*]} )"
        PYTHON_USEDEP=${optflags// /,}

        PYTHON_DEPS=
        local i
        for i in ${PYTHON_COMPAT[@]}; do
                local d
                case ${i} in
                        python*)
                                d='dev-lang/python';;
                        jython*)
                                d='dev-java/jython';;
                        pypy*)
                                d='dev-python/pypy';;
                        *)
                                die "Invalid implementation: ${i}"
                esac

                local v=${i##*[a-z]}
                local usestr
                [[ ${PYTHON_REQ_USE} ]] && usestr="[${PYTHON_REQ_USE}]"
                PYTHON_DEPS+=" python_targets_${i}? (
                        ${d}:${v/_/.}${usestr} )"
        done
}
_python_set_globals

# @FUNCTION: python_get_PYTHON
# @USAGE: <impl>
# @DESCRIPTION:
# Get the Python executable name for the given implementation. Please
# note that this this function returns the 'basename' only.
# If using it for a hashbang, please use #!/usr/bin/env.
python_get_PYTHON() {
        debug-print-function ${FUNCNAME} "${@}"

        local impl=${1/_/.}
        local ret

        case "${impl}" in
                python*|jython*)
                        ret=${impl}
                        ;;
                pypy*)
                        ret=pypy-c${impl#pypy}
                        ;;
                *)
                        die "Invalid argument to python_get_PYTHON: ${1}"
                        ;;
        esac

        debug-print "${FUNCNAME}: ${impl} -> ${ret}"
        echo "${ret}"
}

# @FUNCTION: python_copy_sources
# @DESCRIPTION:
# Create a single copy of the package sources (${S}) for each enabled
# Python implementation.
python_copy_sources() {
        debug-print-function ${FUNCNAME} "${@}"

        local impl
        local bdir=${BUILD_DIR:-${S}}

        debug-print "${FUNCNAME}: bdir = ${bdir}"
        einfo "Will copy sources from ${S}"
        # the order is irrelevant here
        for impl in ${PYTHON_COMPAT[@]}; do
                if use python_targets_${impl}
                then
                        local BUILD_DIR=${bdir%%/}-${impl}

                        einfo "${impl}: copying to ${BUILD_DIR}"
                        debug-print "${FUNCNAME}: [${impl}] cp ${S} => 
${BUILD_DIR}"
                        cp -pr "${S}" "${BUILD_DIR}" || die
                fi
        done
}

# @FUNCTION: python_foreach_impl
# @USAGE: <command> [<args>...]
# @DESCRIPTION:
# Run the given command for each of the enabled Python implementations.
# If additional parameters are passed, they will be passed through
# to the command. If the command fails, python_foreach_impl dies.
# If necessary, use ':' to force a successful return.
#
# Before the command is run, EPYTHON is set to the name of the current
# Python implementation, PYTHON is set to the correct Python executable
# name and exported, and BUILD_DIR is set to a 'default' build directory
# for given implementation (e.g. ${BUILD_DIR:-${S}}-python2_7).
#
# The command is run inside the build directory. If it doesn't exist
# yet, it is created (as an empty directory!). If your build system does
# not support out-of-source builds, you will likely want to use
# python_copy_sources first.
python_foreach_impl() {
        debug-print-function ${FUNCNAME} "${@}"

        local impl
        local bdir=${BUILD_DIR:-${S}}

        debug-print "${FUNCNAME}: bdir = ${bdir}"
        for impl in ${_PYTHON_ALL_IMPLS[@]}; do
                if has ${impl} ${PYTHON_COMPAT[@]} && use python_targets_${impl}
                then
                        local PYTHON=$(python_get_PYTHON "${impl}")
                        local EPYTHON=${PYTHON}
                        local BUILD_DIR=${bdir%%/}-${impl}
                        export PYTHON

                        debug-print "${FUNCNAME}: [${impl}] build_dir = 
${BUILD_DIR}"

                        mkdir -p "${BUILD_DIR}" || die
                        pushd "${BUILD_DIR}" &>/dev/null || die
                        einfo "${EPYTHON}: running ${@}"
                        "${@}" || die "${EPYTHON}: ${1} failed"
                        popd &>/dev/null || die
                fi
        done
}
# Copyright 1999-2012 Gentoo Foundation
# Distributed under the terms of the GNU General Public License v2
# $Header: $

# @ECLASS: distutils-r1
# @MAINTAINER:
# Python herd <pyt...@gentoo.org>
# @AUTHOR:
# Author: Michał Górny <mgo...@gentoo.org>
# Based on the work of: Krzysztof Pawlik <nelch...@gentoo.org>
# @BLURB: A simple eclass to build Python packages using distutils.
# @DESCRIPTION:
# A simple eclass providing functions to build Python packages using
# the distutils build system. It exports phase functions for all
# the src_* phases. Each of the phases runs two pseudo-phases:
# python_..._all() (e.g. python_prepare_all()) once in ${S}, then
# python_...() (e.g. python_prepare()) for each implementation
# (see: python_foreach_impl() in python-r1).
#
# In distutils-r1_src_prepare(), the 'all' function is run before
# per-implementation ones (because it creates the implementations),
# per-implementation functions are run in a random order.
#
# In remaining phase functions, the per-implementation functions are run
# before the 'all' one, and they are ordered from the least to the most
# preferred implementation (so that 'better' files overwrite 'worse'
# ones).
#
# If the ebuild doesn't specify a particular pseudo-phase function,
# the default one will be used (distutils-r1_...). Defaults are provided
# for all per-implementation pseudo-phases, python_prepare_all()
# and python_install_all(); whenever writing your own pseudo-phase
# functions, you should consider calling the defaults (and especially
# distutils-r1_python_prepare_all).
#
# Please note that distutils-r1 sets RDEPEND and DEPEND for you.

case "${EAPI}" in
        0|1|2|3)
                die "Unsupported EAPI=${EAPI} (too old) for ${ECLASS}"
                ;;
        4)
                ;;
        *)
                die "Unsupported EAPI=${EAPI} (unknown) for ${ECLASS}"
                ;;
esac

inherit eutils python-r1

EXPORT_FUNCTIONS src_prepare src_configure src_compile src_test src_install

RDEPEND="${PYTHON_DEPS}
        dev-python/python-exec"
DEPEND=${PYTHON_DEPS}

# @ECLASS-VARIABLE: DOCS
# @DEFAULT_UNSET
# @DESCRIPTION:
# Array containing documents installed using dodoc.
#
# If unset, the default filename list (from PMS) will be used.
#
# Example:
# @CODE
# DOCS=( NEWS README )
# @CODE

# @ECLASS-VARIABLE: HTML_DOCS
# @DEFAULT_UNSET
# @DESCRIPTION:
# Array containing documents installed using dohtml.
#
# Example:
# @CODE
# HTML_DOCS=( doc/html/ )
# @CODE

# @FUNCTION: distutils-r1_python_prepare_all
# @DESCRIPTION:
# The default python_prepare_all(). It applies the patches from PATCHES
# array, then user patches and finally calls python_copy_sources to
# create copies of resulting sources for each Python implementation.
distutils-r1_python_prepare_all() {
        debug-print-function ${FUNCNAME} "${@}"

        [[ ${PATCHES} ]] && epatch "${PATCHES[@]}"

        epatch_user

        # create source copies for each implementation
        python_copy_sources
}

# @FUNCTION: distutils-r1_python_prepare
# @DESCRIPTION:
# The default python_prepare(). Currently it is a no-op
# but in the future it may apply implementation-specific quirks
# to the build system.
distutils-r1_python_prepare() {
        debug-print-function ${FUNCNAME} "${@}"

        :
}

# @FUNCTION: distutils-r1_python_configure
# @DESCRIPTION:
# The default python_configure(). Currently a no-op.
distutils-r1_python_configure() {
        debug-print-function ${FUNCNAME} "${@}"

        :
}

# @FUNCTION: distutils-r1_python_compile
# @USAGE: [additional-args...]
# @DESCRIPTION:
# The default python_compile(). Runs 'setup.py build' using the correct
# Python implementation. Any parameters passed to this function will be
# passed to setup.py.
distutils-r1_python_compile() {
        debug-print-function ${FUNCNAME} "${@}"

        cd "${BUILD_DIR}" || die
        set -- "${PYTHON}" setup.py build "${@}"
        echo "${@}"
        "${@}" || die
}

# @FUNCTION: distutils-r1_python_test
# @DESCRIPTION:
# The default python_test(). Currently a no-op.
distutils-r1_python_test() {
        debug-print-function ${FUNCNAME} "${@}"

        :
}

# @FUNCTION: distutils-r1_rename_scripts
# @DESCRIPTION:
# Renames installed Python scripts to be implementation-suffixed.
# ${PYTHON} has to be set to the expected Python executable (which
# hashbang will be grepped for), and ${EPYTHON} to the implementation
# name (for new name).
distutils-r1_rename_scripts() {
        debug-print-function ${FUNCNAME} "${@}"

        local f
        # XXX: change this if we ever allow directories in bin/sbin
        for f in "${D}"/{bin,sbin,usr/bin,usr/sbin}/*; do
                if [[ -x ${f} ]]; then
                        debug-print "${FUNCNAME}: found executable at 
${f#${D}/}"

                        if [[ "$(head -n 1 "${f}")" == '#!'*${PYTHON}* ]]
                        then
                                debug-print "${FUNCNAME}: matching shebang: 
$(head -n 1 "${f}")"

                                local newf=${f}-${EPYTHON}
                                debug-print "${FUNCNAME}: renamed to 
${newf#${D}/}"
                                mv "${f}" "${newf}" || die
                        fi
                fi
        done
}

# @FUNCTION: distutils-r1_python_install
# @USAGE: [additional-args...]
# @DESCRIPTION:
# The default python_install(). Runs 'setup.py install' using
# the correct Python implementation, and appending the optimization
# flags. Then calls distutils-r1_rename_scripts. Any parameters passed
# to this function will be passed to setup.py.
distutils-r1_python_install() {
        debug-print-function ${FUNCNAME} "${@}"

        local flags

        case "${EPYTHON}" in
                jython*)
                        flags='--compile';;
                *)
                        flags='--compile -O2';;
        esac
        debug-print "${FUNCNAME}: [${EPYTHON}] flags: ${flags}"

        unset PYTHONDONTWRITEBYTECODE

        cd "${BUILD_DIR}" || die
        set -- "${PYTHON}" setup.py install ${flags} --root="${D}" "${@}"
        echo "${@}"
        "${@}" || die

        distutils-r1_rename_scripts
}

# @FUNCTION: distutils-r1_python_install_all
# @DESCRIPTION:
# The default python_install_all(). It symlinks wrappers
# for the implementation-suffixed executables and installs
# documentation.
distutils-r1_python_install_all() {
        debug-print-function ${FUNCNAME} "${@}"

        if declare -p DOCS &>/dev/null; then
                dodoc "${DOCS[@]}" || die "dodoc failed"
        else
                local f
                # same list as in PMS
                for f in README* ChangeLog AUTHORS NEWS TODO CHANGES \
                                THANKS BUGS FAQ CREDITS CHANGELOG; do
                        if [[ -s ${f} ]]; then
                                dodoc "${f}" || die "(default) dodoc ${f} 
failed"
                        fi
                done
        fi

        if declare -p HTML_DOCS &>/dev/null; then
                dohtml -r "${HTML_DOCS[@]}" || die "dohtml failed"
        fi

        # note: keep in sync with ...rename_scripts()
        # also, we assume that each script is installed for all impls
        local any_impl=$(python_get_PYTHON ${PYTHON_COMPAT[0]})
        for f in "${D}"/{bin,sbin,usr/bin,usr/sbin}/*-${any_impl}; do
                if [[ -x ${f} ]]; then
                        debug-print "${FUNCNAME}: found executable at 
${f#${D}/}"

                        local wrapf=${f%-${any_impl}}
                        debug-print "${FUNCNAME}: will link wrapper to 
${wrapf#${D}/}"
                        local wrapfrom
                        case "${f#${D}/}" in
                                usr/bin*)
                                        wrapfrom=
                                        ;;
                                usr/sbin*)
                                        wrapfrom=../bin/
                                        ;;
                                *)
                                        wrapfrom=../usr/bin/
                                        ;;
                        esac
                        debug-print "${FUNCNAME}: (from ${wrapfrom}python-exec)"

                        ln -s ${wrapfrom}python-exec "${wrapf}" || die
                fi
        done
}

distutils-r1_src_prepare() {
        debug-print-function ${FUNCNAME} "${@}"

        # common preparations
        if declare -f python_prepare_all >/dev/null; then
                python_prepare_all
        else
                distutils-r1_python_prepare_all
        fi

        if declare -f python_prepare >/dev/null; then
                python_foreach_impl python_prepare
        else
                distutils-r1_python_prepare
        fi
}

distutils-r1_src_configure() {
        if declare -f python_configure >/dev/null; then
                python_foreach_impl python_configure
        else
                distutils-r1_python_configure
        fi

        if declare -f python_configure_all >/dev/null; then
                python_configure_all
        fi
}

distutils-r1_src_compile() {
        debug-print-function ${FUNCNAME} "${@}"

        if declare -f python_compile >/dev/null; then
                python_foreach_impl python_compile
        else
                python_foreach_impl distutils-r1_python_compile
        fi

        if declare -f python_compile_all >/dev/null; then
                python_compile_all
        fi
}

distutils-r1_src_test() {
        debug-print-function ${FUNCNAME} "${@}"

        if declare -f python_test >/dev/null; then
                python_foreach_impl python_test
        else
                distutils-r1_python_test
        fi

        if declare -f python_test_all >/dev/null; then
                python_test_all
        fi
}

distutils-r1_src_install() {
        debug-print-function ${FUNCNAME} "${@}"

        if declare -f python_install >/dev/null; then
                python_foreach_impl python_install
        else
                python_foreach_impl distutils-r1_python_install
        fi

        if declare -f python_install_all >/dev/null; then
                python_install_all
        else
                distutils-r1_python_install_all
        fi
}

Attachment: signature.asc
Description: PGP signature

Reply via email to