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

Use meson instead of configure for conda install #36489

Closed
wants to merge 19 commits into from
Closed
Show file tree
Hide file tree
Changes from 13 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
26 changes: 17 additions & 9 deletions .github/workflows/ci-conda.yml
Original file line number Diff line number Diff line change
Expand Up @@ -66,21 +66,20 @@ jobs:
conda info
conda list

- name: Configure
- name: Bootstrap
shell: bash -l {0}
continue-on-error: true
run: |
./bootstrap
echo "::add-matcher::.github/workflows/configure-systempackage-problem-matcher.json"
./configure --enable-build-as-root --with-python=$CONDA_PREFIX/bin/python --prefix=$CONDA_PREFIX --enable-system-site-packages $(for pkg in $(./sage -package list :standard: --has-file spkg-configure.m4 --has-file distros/conda.txt --exclude rpy2); do echo --with-system-$pkg=force; done)
echo "::remove-matcher owner=configure-system-package-warning::"
echo "::remove-matcher owner=configure-system-package-error::"
run: ./bootstrap

- name: Build
shell: bash -l {0}
run: |
# Use --no-deps and pip check below to verify that all necessary dependencies are installed via conda.
pip install --no-build-isolation --no-deps -v -v -e ./pkgs/sage-conf ./pkgs/sage-setup
echo "::group::sage-setup"
pip install --no-build-isolation --no-deps -v -v -e ./pkgs/sage-setup
echo "::endgroup::"
echo "::group::sage-conf"
pip install --no-build-isolation --no-deps -v -v -e ./pkgs/sage-conf_meson
echo "::endgroup::"
pip install --no-build-isolation --no-deps --config-settings editable_mode=compat -v -v -e ./src
env:
SAGE_NUM_THREADS: 2
Expand All @@ -95,6 +94,15 @@ jobs:
shell: bash -l {0}
run: ./sage -t --all -p0

# We keep this step for now to make sure that the configure-based setup still works.
- name: Test configure
shell: bash -l {0}
run: |
echo "::add-matcher::.github/workflows/configure-systempackage-problem-matcher.json"
./configure --enable-build-as-root --with-python=$CONDA_PREFIX/bin/python --prefix=$CONDA_PREFIX --enable-system-site-packages $(for pkg in $(./sage -package list :standard: --has-file spkg-configure.m4 --has-file distros/conda.txt --exclude rpy2); do echo --with-system-$pkg=force; done)
echo "::remove-matcher owner=configure-system-package-warning::"
echo "::remove-matcher owner=configure-system-package-error::"

- name: Print logs
if: always()
run: |
Expand Down
3 changes: 2 additions & 1 deletion .vscode/settings.json
Original file line number Diff line number Diff line change
Expand Up @@ -34,5 +34,6 @@
"sagemath",
"Cython"
],
"editor.formatOnType": true
"editor.formatOnType": true,
"esbonio.sphinx.confDir": ""
Copy link
Contributor

Choose a reason for hiding this comment

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

What does this change do? Does it have a relation to the goal of this PR?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Vscode is adding this every time I'm opening the project. I was bound to commit it at some point ;-). I doesn't hurt...

}
1 change: 1 addition & 0 deletions bootstrap-conda
Original file line number Diff line number Diff line change
Expand Up @@ -98,6 +98,7 @@ echo >&2 $0:$LINENO: generate conda environment files
) > environment-template.yml
(
sed 's/name: sage-build/name: sage/' environment-template.yml
echo " - meson"
echo " # Additional packages providing all dependencies for the Sage library"
for pkg in $SAGELIB_SYSTEM_PACKAGES; do
echo " - $pkg"
Expand Down
40 changes: 40 additions & 0 deletions meson.build
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
project('sage', 'c')
Copy link
Contributor

Choose a reason for hiding this comment

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

I'd be more comfortable with this file being local to sage-conf_meson, so as not to create expectations that Sage-the-distribution will switch to meson

Copy link
Contributor Author

Choose a reason for hiding this comment

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

The idea in the middle to long term is to support running meson compile from the root directory to compile sagelib with meson (using conda). For this the meson build file needs to be there (and its pretty much convention to have such a build file in the root). Sage-the-distro supporting meson would mean that there is a meson file in the build directory, or not?

Copy link
Contributor

Choose a reason for hiding this comment

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

I'm not aware of such a plan, and I'd suggest that if you have a plan to use a new Issue in which you explain it.

From what you wrote above, it seems unlikely that it would be the same meson.build file that would be suitable for both purposes, so nothing is gained by putting it there.

Copy link
Contributor Author

@tobiasdiez tobiasdiez Oct 22, 2023

Choose a reason for hiding this comment

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

It's part of #34630. In the long-term you want to use of course meson-python (configured via pyproject.toml). But in the short to midterm its easier to just call meson directly. That's at least how scipy/numpy proceeded: https://github.com/scipy/scipy/pull/14847/files#diff-e42998d51257e6f84aa51ffa5f4ed1544cb12f97a94978c44f3bc281b5324d79R390-R500

And yes, the meson file added in this PR can be used for this purpose. In fact, I believe we can get ride of the conf and setup packages, but of course there still needs to be some way to provide the runtime-config (probably by just putting the generated conf file in the correct place in the builddir).

Copy link
Contributor

Choose a reason for hiding this comment

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

Well, #34630 has nothing to with stuff that belongs in SAGE_ROOT.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

So it's for compiling the Sage library, for the single use case of the all-conda install?

There is nothing conda-specific about the meson files, it's only that conda provides a nice fixed environment and thus is a good starting point. You can try to run meson setup outside of a conda environment and it should work (if you have all the necessary packages installed in the system).

Copy link
Contributor

Choose a reason for hiding this comment

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

I'll explain and summarize once more:

  • What you just described ("it should work (if you have all the necessary packages installed in the system).") means that you are configuring / building the Sage library.
  • This is different from what our configure.ac and Makefile do. They configure / build the Sage distribution.
  • There is no plan to switch the Sage distribution to using meson.
  • When people see a file meson.build next to configure.ac and Makefile, they are bound to think that using meson is an alternative for building the same thing. So this has the potential to mislead.

Copy link
Contributor Author

@tobiasdiez tobiasdiez Oct 29, 2023

Choose a reason for hiding this comment

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

  • When people see a file meson.build next to configure.ac and Makefile, they are bound to think that using meson is an alternative for building the same thing. So this has the potential to mislead.

So if you see a hammer and a saw next to each other you are bound to see them as alternatives for building the same thing?

Conceptually, meson setup does the same thing as configure: go through the declared dependencies and write out a config file. Similarly, meson compile will do the same thing as make sage-all (or whatever the correct target is for building sagelib).

It's also not technically possible to move the meson file in the root to src since meson's subdir only walks down the tree.

Anyway, I'm tired of this stupid discussion and will stop participating in it. If you think that the harm created by the meson file in the root outweighs the positives of this PR, then please go ahead and continue blocking it. Please leave the "ready for review" tag on, as there are no work items that referee and PR creator agree on (it just that you don't like a small design choice in the PR).

Copy link
Contributor

Choose a reason for hiding this comment

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

Similarly, meson compile will do the same thing as make sage-all (or whatever the correct target is for building sagelib).

The target make sage-all does not exist. make all, the default target, builds the Sage distribution.

Copy link
Contributor

Choose a reason for hiding this comment

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

Sorry, but there are a lot of files in the root folder that are not related to the distribution.
[...] tox ini [...]

Actually SAGE_ROOT/tox.ini defines the portability tests of the Sage distribution.

cc = meson.get_compiler('c')
conf_data = configuration_data()

conf_data.set('PACKAGE_VERSION', '1.2.3')

maxima = find_program('maxima', required: true)
conf_data.set('SAGE_MAXIMA', maxima.path())
# Conda's ecl does not have any problems with Maxima, so nothing needs to be set here:
conf_data.set('SAGE_MAXIMA_FAS', '')

# Kenzo cannot yet be provided by the system, so we always use the SAGE_LOCAL path for now.
conf_data.set('SAGE_KENZO_FAS', '\'${prefix}\'/lib/ecl/kenzo.fas')

conf_data.set('SAGE_ARB_LIBRARY', 'arb')

#ntl = dependency('ntl', required: true) doesn't work, so ask the compiler directly:
ntl = cc.find_library('ntl', required: true)
# It can be found, so we don't have to set anything here:
conf_data.set('NTL_INCDIR', '')
conf_data.set('NTL_LIBDIR', '')

ecl_config = find_program('ecl-config', required: true)
conf_data.set('SAGE_ECL_CONFIG', ecl_config.path())

conf_data.set('SAGE_ARCHFLAGS', 'unset')

# not needed when using conda, as we then don't build any pc files
conf_data.set('SAGE_PKG_CONFIG_PATH', '')

openmp = dependency('openmp', required : false)
if openmp.found()
conf_data.set('OPENMP_CFLAGS', '-fopenmp')
conf_data.set('OPENMP_CXXFLAGS', '-fopenmp')
endif

configure_file(input : 'pkgs/sage-conf_conda/_sage_conf/_conf.py.in',
output : '_conf.py',
configuration : conf_data)

7 changes: 7 additions & 0 deletions pkgs/sage-conf/pyproject.toml
Original file line number Diff line number Diff line change
@@ -1,3 +1,10 @@
[build-system]
requires = ["setuptools", "wheel"]
build-backend = "setuptools.build_meta"

[project]
name = "sage-conf"
dynamic = ["version"]

[tool.setuptools.dynamic]
version = {file = ["VERSION.txt"]}
tobiasdiez marked this conversation as resolved.
Show resolved Hide resolved
7 changes: 7 additions & 0 deletions pkgs/sage-conf_meson/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
/_sage_conf/_conf.py
/builddir
/build
/dist
/*.egg-info
/.tox
/bin/sage-env-config
1 change: 1 addition & 0 deletions pkgs/sage-conf_meson/README.rst
1 change: 1 addition & 0 deletions pkgs/sage-conf_meson/VERSION.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
10.2.beta7
1 change: 1 addition & 0 deletions pkgs/sage-conf_meson/_sage_conf
1 change: 1 addition & 0 deletions pkgs/sage-conf_meson/pyproject.toml
6 changes: 6 additions & 0 deletions pkgs/sage-conf_meson/sage_conf.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
from _sage_conf.__main__ import _main

from builddir._conf import *
from builddir.build_info import CONDA_PREFIX, SAGE_ROOT
Copy link
Contributor

Choose a reason for hiding this comment

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

This "from builddir... import" makes me a bit uncomfortable.
This seems to assume that . is in sys.path
Please test that also non-editable installs (via wheels) of sage-conf_meson work.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

The builddir here is generated under pkgs/sage-conf_meson

Copy link
Contributor

Choose a reason for hiding this comment

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

Yes, I knew that when I wrote the above. So?

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 see the difference between configure creating the _conf.py file and here we are using meson to create the builddir folder. Can you please be more specific what problem you encounter?

Copy link
Contributor

Choose a reason for hiding this comment

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

I don't see the difference

In https://github.com/sagemath/sage/blob/develop/pkgs/sage-conf/sage_conf.py#L1 we have:
from _sage_conf._conf import *

_sage_conf is an installed top-level package.

Here: from builddir._conf import *

builddir is not an installed top-level package.

More questions?

Copy link
Contributor

Choose a reason for hiding this comment

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

At this time, I don't have a comment on what variables provided by sage_conf are needed at the runtime of the Sage library.

That's my observation: apparently none. The fallbacks are good enough for conda.

You may be right about this, but a lot currently still depends on the assumption "$SAGE_LOCAL = $CONDA_PREFIX" (e.g., see #30914 for pointers), which is guaranteed by the all-conda installation instructions, but I am not sure that we will want to keep it in the future.

A problem with this assumption is that we currently have an inflexible either-or situation:

  • We either take all packages from conda and do not use the Sage build system at all, for anything;
  • or we use the Sage build system for everything and then cannot use Python packages from conda. (For the purpose of this discussion, let's ignore the --enable-system-site-packages switch.)

Copy link
Contributor

Choose a reason for hiding this comment

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

sage-conf is a declared install-requires of sagelib, so some version of it needs to be installed.

So can I change that to an optional dependency? (since it seems to be optional)

No, by design it's required, but as is explained in https://github.com/sagemath/sage/blob/develop/pkgs/sage-conf/README.rst, you can pick your own implementation.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

At this time, I don't have a comment on what variables provided by sage_conf are needed at the runtime of the Sage library.

That's my observation: apparently none. The fallbacks are good enough for conda.

You may be right about this, but a lot currently still depends on the assumption "$SAGE_LOCAL = $CONDA_PREFIX" (e.g., see #30914 for pointers), which is guaranteed by the all-conda installation instructions, but I am not sure that we will want to keep it in the future.

A problem with this assumption is that we currently have an inflexible either-or situation:

* We either take all packages from conda and do not use the Sage build system at all, for anything;

* or we use the Sage build system for everything and then cannot use Python packages from conda. (For the purpose of this discussion, let's ignore the `--enable-system-site-packages` switch.)

Right, but in the "we take everything from conda" the assumption will always be valid, right? So in this case we don't need a sage_conf package. If in the future some semi-mixed versions are supported, then this method needs its own sage_conf package.

So can I change that to an optional dependency? (since it seems to be optional)

No, by design it's required, but as is explained in https://github.com/sagemath/sage/blob/develop/pkgs/sage-conf/README.rst, you can pick your own implementation.

This document doesn't say anything on why we need the package nor why the design cannot be changed.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Either way, I've fixed the package now. Please give it a try.

Copy link
Contributor

Choose a reason for hiding this comment

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

Right, but in the "we take everything from conda" the assumption will always be valid, right?

My point is that this dichotomy of the two installation methods is not sustainable; so it would be unwise to be invested in a "pure" mode that seeks to eliminate sage_conf altogether.


SAGE_LOCAL = CONDA_PREFIX
62 changes: 62 additions & 0 deletions pkgs/sage-conf_meson/setup.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,62 @@
import os
import sys
from distutils.command.build_scripts import \
build_scripts as distutils_build_scripts
from pathlib import Path

from setuptools import setup
from setuptools.command.build_py import build_py as setuptools_build_py
from setuptools.command.editable_wheel import \
editable_wheel as setuptools_editable_wheel
from setuptools.errors import SetupError


class build_py(setuptools_build_py):
def run(self):
here = Path(__file__).parent
if self.editable_mode:
root = here.parent.parent
else:
raise SetupError('Not supported')

conda_prefix = os.environ.get('CONDA_PREFIX', '')
if not conda_prefix:
raise SetupError(
'No conda environment is active. '
'See https://doc.sagemath.org/html/en/installation/conda.html on how to get started.'
)

builddir = here / "builddir"
cmd = f"cd {root} && meson setup {builddir} --wipe"
print(f"Running {cmd}")
sys.stdout.flush()
if os.system(cmd) != 0:
raise SetupError("configure failed")

# Write build info
with open(builddir / 'build_info.py', 'w', encoding="utf-8") as build_info:
build_info.write(f'SAGE_ROOT = "{root}"\n')
build_info.write(f'CONDA_PREFIX = "{conda_prefix}"\n')


class build_scripts(distutils_build_scripts):
def run(self):
self.distribution.scripts.append(os.path.join('bin', 'sage-env-config'))
if not self.distribution.entry_points:
self.entry_points = self.distribution.entry_points = dict()
distutils_build_scripts.run(self)


class editable_wheel(setuptools_editable_wheel):
r"""
Customized so that exceptions raised by our build_py
do not lead to the "Customization incompatible with editable install" message
"""
_safely_run = setuptools_editable_wheel.run_command


setup(
cmdclass=dict(
build_py=build_py, build_scripts=build_scripts, editable_wheel=editable_wheel
)
)
5 changes: 4 additions & 1 deletion src/doc/en/installation/conda.rst
Original file line number Diff line number Diff line change
Expand Up @@ -139,9 +139,12 @@ Here we assume that you are using a git checkout.
- Bootstrap the source tree and install the build prerequisites and the Sage library::

$ ./bootstrap
$ pip install --no-build-isolation -v -v --editable ./pkgs/sage-conf_conda ./pkgs/sage-setup
$ pip install --no-build-isolation -v -v --editable ./pkgs/sage-conf_meson ./pkgs/sage-setup
$ pip install --no-build-isolation --config-settings editable_mode=compat -v -v --editable ./src

In case of errors, try to use ``sage-conf_conda`` instead of ``sage-conf_meson``,
and please report the problem by opening an issue on GitHub.

- Verify that Sage has been installed::

$ sage -c 'print(version())'
Expand Down
Loading