Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Better integration of setup.py to invoke b2 #5325

Merged
merged 1 commit into from
Dec 20, 2020

Conversation

AllSeeingEyeTolledEweSew
Copy link
Contributor

Hi,

I rewrote setup.py in the python bindings to be better-integrated with the b2 build system.

My use case is that I want to more easily build wheels of libtorrent. Specifically, I want to use tox and pre-commit in my libtorrent-based python project. These (and other tools) set up isolated environments and try to install dependencies, and since libtorrent isn't published on pypi, the easiest way to make these tools work is run a local pypi instance which provides access to libtorrent.

It was possible to make wheels before, but this integration fix makes it much easier. I now can build wheels with custom b2 compilation, like so:

$ python setup.py build_ext -j9 --debug --libtorrent-link=static bdist_wheel

I deleted all the existing build infrastructure from setup.py as I don't believe it was very useful: the --bjam mode didn't seem to actually integrate with distutils. It just invoked b2 from the global level, but didn't allow for passing options. The non---bjam mode seemed very limited and wasn't referenced at all in the build docs.

I can't find any reference that setup.py is actually used by anyone today; debian and alpine's build scripts just invoke autotools and don't use setup.py. This is appropriate for them, but setuptools is a better choice for packaging in the distro-agnostic python world.

@AllSeeingEyeTolledEweSew
Copy link
Contributor Author

AllSeeingEyeTolledEweSew commented Dec 2, 2020

I'm not sure what the relationship should be of this PR to #4218; it needs some discussion with the author.

@arvidn
Copy link
Owner

arvidn commented Dec 3, 2020

it would be great if you could extend the GH action for building the python binding to use this, and maybe even upload the package as an artifact

@AllSeeingEyeTolledEweSew
Copy link
Contributor Author

it would be great if you could extend the GH action for building the python binding to use this, and maybe even upload the package as an artifact

Interesting. I'd love to achieve the same goals as #4218, but it's a big PR, and I'd at least like to split out the artifact building part. I think it's going to get hairy.

Given that you have this:

if <boost-link>static in $(properties) && $(BOOST_VERSION_TAG) < 1_74 && <target-os>linux in $(properties)
{
ECHO "WARNING: you cannot link statically against boost-python on linux before version 1.74.0, because it links against pthread statically in that case, which is not allowed" ;
}

If this is true (though the boost-python docs don't mention it), then to build an appropriate linux wheel we need boost-python 1.74, which isn't in ubuntu yet, so we need a local download-and-install script as in #4218 . It just seems a bit much for this PR

@arvidn
Copy link
Owner

arvidn commented Dec 3, 2020

sure, I don't want to bloat this PR. But, if this code you wrote isn't running on CI, how do I know it works?

run: |
cd bindings/python
b2 -j2 stage_module stage_dependencies cxxstd=11
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

the build is failing be cause cxxstd=11 isn't passed to b2 right now it seems

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

oh, is cxxstd=11 required on macos right now?

setup.py could detect and pass this, but should the jamfile enforce it instead? 🤔

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

in later version (RC_2_0 and master) the Jamfile defaults to cxxstd=14 and cxxstd=17 respectively. I just haven't made that change to RC_1_2 because it supports being built with an old version of boost where the cxxstd option didn't exist yet

run: |
sudo apt install libboost-tools-dev libboost-python-dev libboost-dev libboost-system-dev
echo "using gcc ;" >>~/user-config.jam
echo "using python ;" >>~/user-config.jam
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is there a good reason not to configure boost build? It seems risky to rely on its default heuristic to pick gcc

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This was intentional; the typical expectation with setuptools is that it should be able to pick sane defaults without special configuration (except for installing dependencies to build extensions).

I could add a separate test to pass toolset=gcc. I'm still trying to get a feel for what other specific feature properties to test.

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would like to preserve the test of b2 in the bindings/python directory. would you mind restoring this and adding another job that runs setup.py?

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would like to keep this python job. would you mind just reverting this file?

run: |
brew install boost-build boost boost-python3 python@3.9
export PYTHON_INCLUDE=`echo /usr/local/Cellar/python@3.9/*/Frameworks/Python.framework/Versions/3.9/include/python3.9` ;
echo "using darwin ;" >>~/user-config.jam
echo "using python : 3.9 : /usr/local/opt/python@3.9/bin/python3 : $(PYTHON_INCLUDE) : /usr/local/opt/python@3.9/Frameworks/Python.framework/Versions/3.9/lib/python3.9/config-3.9-darwin : <target-os>darwin ;" >> ~/user-config.jam
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Are you confident boost build will find the correct python headers and library without configuring it here?

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

in fact, it looks like it's building against python 2.6 in the github action now, since you removed this configuration step

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm confident that I want this to be the test :)

Another expectation with setuptools is that it builds the package for the running interpreter, so ideally setup.py or the jamfile should be able to work out these paths. I don't have that much experience with macos though, so I'm not sure if this expectation is reasonable.

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ok, makes sense. I would also like to test that the python binding builds using b2 using the boost build user configuration.

@arvidn
Copy link
Owner

arvidn commented Dec 7, 2020

I think this error looks like it's caused by setuptools trying to build the extension. That's not the intention, right? it should use the binary built by b2

@AllSeeingEyeTolledEweSew
Copy link
Contributor Author

I think this error looks like it's caused by setuptools trying to build the extension. That's not the intention, right? it should use the binary built by b2

Yes. Looks like I actually needed to pass the extension suffix to boost so its output is recognized by setuptools; the debian version shadowed this problem.

I'm doing most of my testing on a debian-based arch and I'm finding that debian has patched the daylights out of python.jam from boost-build, with prety significant behavior changes. It makes me think that all the CI should be run on non-ubuntu linux as well as ubuntu, or else install boost from source.

@AllSeeingEyeTolledEweSew AllSeeingEyeTolledEweSew force-pushed the asetes-setup.py-b2 branch 3 times, most recently from 56245dc to c6a2c56 Compare December 7, 2020 09:51
@AllSeeingEyeTolledEweSew
Copy link
Contributor Author

I made setup.py default to passing cxxstd=11, only on osx. Hopefully this is appropriate.

I put setup.py invocation on osx in a separate github action, which is passing now.

It's not currently invoked on windows. Should I create a separate appveyor matrix entry to invoke setup.py, separate from one that builds the binding with b2? Appveyor already takes forever.

I doubt the linux CI failure is a result of this PR.

@AllSeeingEyeTolledEweSew
Copy link
Contributor Author

I rebased to make sure the CI failure wasn't mine.

@arvidn could you advise on whether you'd like to use appveyor or github actions for new windows CI?

@FranciscoPombal
Copy link
Contributor

@AllSeeingEyeTolledEweSew

I rebased to make sure the CI failure wasn't mine.

@arvidn could you advise on whether you'd like to use appveyor or github actions for new windows CI?

I was working on GH Actions CI for Windows + CMake here #5197, I should finish it relatively soon.

cd bindings/python
/usr/local/opt/python@3.9/bin/python3.9 test.py

python_b2:
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it looks like you preserved the existing test here. would you mind re-ordering these jobs to make the diff smaller, and reflect the fact that the existing test is unchanged?

@arvidn
Copy link
Owner

arvidn commented Dec 10, 2020

lgtm, except the github workflow changes

@AllSeeingEyeTolledEweSew AllSeeingEyeTolledEweSew force-pushed the asetes-setup.py-b2 branch 4 times, most recently from 10d30fd to ef1e2c0 Compare December 16, 2020 20:30
- name: install dependencies (linux)
if: runner.os == 'Linux'
run: |
sudo apt install libboost-tools-dev libboost-python-dev libboost-dev libboost-system-dev python3 python3-setuptools
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

you should probably rebase now that #5749 landed, as well as add those sudo apt update here as well

self.pic = None
self.optimization = None
self.hash = None
if platform.system() == "Darwin":
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't think this distinction is important. You just happen to have an older compiler on MacOS that doesn't default to C++11. The more interesting question is, which ABI is python itself use? perhaps C++11 is too old.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't think this distinction is important. You just happen to have an older compiler on MacOS that doesn't default to C++11.

I see. Do you have opinions on whether it would be better to default toolset=gcc on darwin, rather than default cxxstd=11? I'll play with this.

I guess what's really needed is a default of cxxstd=11 if toolset=darwin, but setup.py doesn't know if toolset=darwin will be autodiscovered in b2. IMO would be much better to just apply this in some Jamfile in RC_1_2, rather than here.

The more interesting question is, which ABI is python itself use? perhaps C++11 is too old.

cpython's ABI is C only. So cxxstd's ABI just matters for the extension's downstream dynamic linkage. Ideally, when packaging a wheel for pypi, the choice of cxxstd should never matter to consumers.

(There are various C ABIs defined, but setup.py's scope is to only know about the ABI of the running interpreter, and configure boost to build for that)

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

IMO would be much better to just apply this in some Jamfile in RC_1_2, rather than here.

It is in RC_2_0 and later

cpython's ABI is C only. So cxxstd's ABI just matters for the extension's downstream dynamic linkage.

Which C++ standard is used affects which C standard the compiler is using as well. The C standard doesn't define an ABI, just like C++ doesn't. It may be easier for platforms to maintain a stable ABI in C, but there's no guarantee. windows/MSVC is a good example.

I imagine specifying stdcxx=11 here is safe, but with MSVC, the version of the compiler depends on the version of python.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In truth I don't think we need to work too hard to solve this here.

I don't think invoking setup.py with no args needs to produce the perfect most-compatible build. When packaging wheels for pypi, I expect we'd do things like --libtorrent-link=static --boost-link=static which wouldn't be appropriate in most other cases.

I just wanted setup.py with no args to produce something useful, both to help users, and to trim down os-specific overrides in CI. I expect the impact of changing this default later will be low.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

For what it's worth, I haven't seen evidence that python is required to be compiled with any particular C standard on mac/linux.

On linux particularly, I struggle to think of any problems that would be caused for an executable linking to a library, if they differed in C standard alone. I would expect versioned symbols in libc to resolve the differences (in a perfect world). I don't have much macos experience so I don't know if they have something similar.

@AllSeeingEyeTolledEweSew
Copy link
Contributor Author

I think it's ready now.

Note that I don't currently have a windows workflow to invoke setup.py, because figuring out how to install boost is daunting. Once #5197 lands, I'll reuse its setup methods to extend my python.yml.

elif "version = '" in line and name.endswith('setup.py'):
line = " version='%d.%d.%d',\n" % (version[0], version[1], version[2])
line = " version=\"%d.%d.%d\",\n" % (version[0], version[1], version[2])
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it looks like this line is duplicated. I should remove one of them once this lands

@arvidn
Copy link
Owner

arvidn commented Dec 19, 2020

great! are these commits squashed to reasonable changes? (if not, please do that and I'll land it afterwards)

@AllSeeingEyeTolledEweSew
Copy link
Contributor Author

great! are these commits squashed to reasonable changes? (if not, please do that and I'll land it afterwards)

I tried to keep them clean, but failed by the end. I squashed to one as it's pretty straightforward.

@arvidn arvidn merged commit b87d012 into arvidn:RC_1_2 Dec 20, 2020
@AllSeeingEyeTolledEweSew AllSeeingEyeTolledEweSew deleted the asetes-setup.py-b2 branch August 2, 2021 08:16
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants