-
-
Notifications
You must be signed in to change notification settings - Fork 1k
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
Better integration of setup.py to invoke b2 #5325
Conversation
I'm not sure what the relationship should be of this PR to #4218; it needs some discussion with the 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: libtorrent/bindings/python/Jamfile Lines 137 to 140 in f726d10
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 |
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? |
c0ff27d
to
db655b0
Compare
.github/workflows/macos.yml
Outdated
run: | | ||
cd bindings/python | ||
b2 -j2 stage_module stage_dependencies cxxstd=11 |
There was a problem hiding this comment.
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
There was a problem hiding this comment.
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? 🤔
There was a problem hiding this comment.
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
.github/workflows/linux.yml
Outdated
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 |
There was a problem hiding this comment.
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
There was a problem hiding this comment.
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.
There was a problem hiding this comment.
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
?
There was a problem hiding this comment.
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?
.github/workflows/macos.yml
Outdated
run: | | ||
brew install boost-build boost boost-python3 [email protected] | ||
export PYTHON_INCLUDE=`echo /usr/local/Cellar/[email protected]/*/Frameworks/Python.framework/Versions/3.9/include/python3.9` ; | ||
echo "using darwin ;" >>~/user-config.jam | ||
echo "using python : 3.9 : /usr/local/opt/[email protected]/bin/python3 : $(PYTHON_INCLUDE) : /usr/local/opt/[email protected]/Frameworks/Python.framework/Versions/3.9/lib/python3.9/config-3.9-darwin : <target-os>darwin ;" >> ~/user-config.jam |
There was a problem hiding this comment.
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?
There was a problem hiding this comment.
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
There was a problem hiding this comment.
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.
There was a problem hiding this comment.
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.
7822727
to
d195538
Compare
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 |
d195538
to
4bcd247
Compare
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 |
56245dc
to
c6a2c56
Compare
I made I put It's not currently invoked on windows. Should I create a separate appveyor matrix entry to invoke I doubt the linux CI failure is a result of this PR. |
282c8b3
to
0a36060
Compare
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? |
.github/workflows/macos.yml
Outdated
cd bindings/python | ||
/usr/local/opt/[email protected]/bin/python3.9 test.py | ||
|
||
python_b2: |
There was a problem hiding this comment.
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?
lgtm, except the github workflow changes |
10d30fd
to
ef1e2c0
Compare
- 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 |
There was a problem hiding this comment.
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": |
There was a problem hiding this comment.
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.
There was a problem hiding this comment.
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)
There was a problem hiding this comment.
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.
There was a problem hiding this comment.
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.
There was a problem hiding this comment.
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.
ef1e2c0
to
3420448
Compare
9c97626
to
f97022c
Compare
I think it's ready now. Note that I don't currently have a windows workflow to invoke |
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]) |
There was a problem hiding this comment.
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
great! are these commits squashed to reasonable changes? (if not, please do that and I'll land it afterwards) |
f97022c
to
de9f8cb
Compare
I tried to keep them clean, but failed by the end. I squashed to one as it's pretty straightforward. |
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:
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 usesetup.py
. This is appropriate for them, but setuptools is a better choice for packaging in the distro-agnostic python world.