mirror of
				https://github.com/f-droid/fdroidserver.git
				synced 2025-11-04 06:30:27 +03:00 
			
		
		
		
	new subcommand "push": podman/vagrant implementation
This commit is contained in:
		
							parent
							
								
									69c67badfb
								
							
						
					
					
						commit
						9fa7193620
					
				
					 4 changed files with 298 additions and 2 deletions
				
			
		| 
						 | 
					@ -67,6 +67,7 @@ COMMANDS = OrderedDict([
 | 
				
			||||||
COMMANDS_INTERNAL = [
 | 
					COMMANDS_INTERNAL = [
 | 
				
			||||||
    "destroy",
 | 
					    "destroy",
 | 
				
			||||||
    "exec",
 | 
					    "exec",
 | 
				
			||||||
 | 
					    "push",
 | 
				
			||||||
    "up",
 | 
					    "up",
 | 
				
			||||||
]
 | 
					]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -4,10 +4,10 @@
 | 
				
			||||||
#
 | 
					#
 | 
				
			||||||
# Copyright (C) 2010-2016, Ciaran Gultnieks, ciaran@ciarang.com
 | 
					# Copyright (C) 2010-2016, Ciaran Gultnieks, ciaran@ciarang.com
 | 
				
			||||||
# Copyright (C) 2013-2017, Daniel Martí <mvdan@mvdan.cc>
 | 
					# Copyright (C) 2013-2017, Daniel Martí <mvdan@mvdan.cc>
 | 
				
			||||||
# Copyright (C) 2013-2021, Hans-Christoph Steiner <hans@eds.org>
 | 
					# Copyright (C) 2013-2025, Hans-Christoph Steiner <hans@eds.org>
 | 
				
			||||||
# Copyright (C) 2017-2018, Torsten Grote <t@grobox.de>
 | 
					# Copyright (C) 2017-2018, Torsten Grote <t@grobox.de>
 | 
				
			||||||
# Copyright (C) 2017, tobiasKaminsky <tobias@kaminsky.me>
 | 
					# Copyright (C) 2017, tobiasKaminsky <tobias@kaminsky.me>
 | 
				
			||||||
# Copyright (C) 2017-2021, Michael Pöhn <michael.poehn@fsfe.org>
 | 
					# Copyright (C) 2017-2025, Michael Pöhn <michael.poehn@fsfe.org>
 | 
				
			||||||
# Copyright (C) 2017,2021, mimi89999 <michel@lebihan.pl>
 | 
					# Copyright (C) 2017,2021, mimi89999 <michel@lebihan.pl>
 | 
				
			||||||
# Copyright (C) 2019-2021, Jochen Sprickerhof <git@jochen.sprickerhof.de>
 | 
					# Copyright (C) 2019-2021, Jochen Sprickerhof <git@jochen.sprickerhof.de>
 | 
				
			||||||
# Copyright (C) 2021, Felix C. Stegerman <flx@obfusk.net>
 | 
					# Copyright (C) 2021, Felix C. Stegerman <flx@obfusk.net>
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										208
									
								
								fdroidserver/push.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										208
									
								
								fdroidserver/push.py
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,208 @@
 | 
				
			||||||
 | 
					#!/usr/bin/env python3
 | 
				
			||||||
 | 
					#
 | 
				
			||||||
 | 
					# push.py - part of the FDroid server tools
 | 
				
			||||||
 | 
					# Copyright (C) 2024-2025, Hans-Christoph Steiner <hans@eds.org>
 | 
				
			||||||
 | 
					# Copyright (C) 2024-2025, Michael Pöhn <michael@poehn.at>
 | 
				
			||||||
 | 
					#
 | 
				
			||||||
 | 
					# This program is free software: you can redistribute it and/or modify
 | 
				
			||||||
 | 
					# it under the terms of the GNU Affero General Public License as published by
 | 
				
			||||||
 | 
					# the Free Software Foundation, either version 3 of the License, or
 | 
				
			||||||
 | 
					# (at your option) any later version.
 | 
				
			||||||
 | 
					#
 | 
				
			||||||
 | 
					# This program is distributed in the hope that it will be useful,
 | 
				
			||||||
 | 
					# but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
				
			||||||
 | 
					# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
				
			||||||
 | 
					# GNU Affero General Public License for more details.
 | 
				
			||||||
 | 
					#
 | 
				
			||||||
 | 
					# You should have received a copy of the GNU Affero General Public License
 | 
				
			||||||
 | 
					# along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					"""Push files into the build setup for a given ApplicationID:versionCode.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					This is a carefully constructed method to copy files from the host
 | 
				
			||||||
 | 
					filesystem to inside the buildserver VM/container to run the build.  The
 | 
				
			||||||
 | 
					source paths are generated based on the build metadata, and the
 | 
				
			||||||
 | 
					destination is forced to build home.  This is not meant as a generic
 | 
				
			||||||
 | 
					method for copying files, that is already provided by each VM/container
 | 
				
			||||||
 | 
					system (e.g. `podman cp`).
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Since this is an internal command, the strings are not localized.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import logging
 | 
				
			||||||
 | 
					import os
 | 
				
			||||||
 | 
					import sys
 | 
				
			||||||
 | 
					import tarfile
 | 
				
			||||||
 | 
					import tempfile
 | 
				
			||||||
 | 
					import traceback
 | 
				
			||||||
 | 
					from argparse import ArgumentParser
 | 
				
			||||||
 | 
					from pathlib import Path
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from . import common, metadata
 | 
				
			||||||
 | 
					from .exception import BuildException
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def podman_push(paths, appid, vercode, as_root=False):
 | 
				
			||||||
 | 
					    """Push relative paths into the podman container using the tar method.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    This builds up a tar file from the supplied paths to send into
 | 
				
			||||||
 | 
					    container via put_archive(). This assumes it is running in the
 | 
				
			||||||
 | 
					    base of fdroiddata and it will push into common.BUILD_HOME in the
 | 
				
			||||||
 | 
					    container.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    This is a version of podman.api.create_tar() that builds up via
 | 
				
			||||||
 | 
					    adding paths rather than giving a base dir and an exclude list.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    """
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def _tar_perms_filter(tarinfo):
 | 
				
			||||||
 | 
					        """Force perms to something safe."""
 | 
				
			||||||
 | 
					        if '__pycache__' in tarinfo.name:
 | 
				
			||||||
 | 
					            return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if as_root:
 | 
				
			||||||
 | 
					            tarinfo.uid = tarinfo.gid = 0
 | 
				
			||||||
 | 
					            tarinfo.uname = tarinfo.gname = 'root'
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            tarinfo.uid = tarinfo.gid = 1000
 | 
				
			||||||
 | 
					            tarinfo.uname = tarinfo.gname = common.BUILD_USER
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if tarinfo.isdir():
 | 
				
			||||||
 | 
					            tarinfo.mode = 0o0755
 | 
				
			||||||
 | 
					        elif tarinfo.isfile():
 | 
				
			||||||
 | 
					            if tarinfo.mode & 0o111:
 | 
				
			||||||
 | 
					                tarinfo.mode = 0o0755
 | 
				
			||||||
 | 
					            else:
 | 
				
			||||||
 | 
					                tarinfo.mode = 0o0644
 | 
				
			||||||
 | 
					        elif not tarinfo.issym():  # symlinks shouldn't need perms set
 | 
				
			||||||
 | 
					            raise BuildException(f'{tarinfo.name} is not a file or directory!')
 | 
				
			||||||
 | 
					        return tarinfo
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if isinstance(paths, Path):
 | 
				
			||||||
 | 
					        paths = [str(paths)]
 | 
				
			||||||
 | 
					    if isinstance(paths, str):
 | 
				
			||||||
 | 
					        paths = [paths]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    container = common.get_podman_container(appid, vercode)
 | 
				
			||||||
 | 
					    with tempfile.TemporaryFile() as tf:
 | 
				
			||||||
 | 
					        with tarfile.TarFile(fileobj=tf, mode='w') as tar:
 | 
				
			||||||
 | 
					            for f in paths:
 | 
				
			||||||
 | 
					                if Path(f).is_absolute():
 | 
				
			||||||
 | 
					                    raise BuildException(f'{f} must be relative to {Path.cwd()}')
 | 
				
			||||||
 | 
					                # throw ValueError on bad path
 | 
				
			||||||
 | 
					                f = (Path.cwd() / f).resolve().relative_to(Path.cwd())
 | 
				
			||||||
 | 
					                tar.add(f, filter=_tar_perms_filter)
 | 
				
			||||||
 | 
					        tf.seek(0)
 | 
				
			||||||
 | 
					        container.put_archive(common.BUILD_HOME, tf.read())
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def vagrant_push(paths, appid, vercode):
 | 
				
			||||||
 | 
					    """Push files into a build specific vagrant VM."""
 | 
				
			||||||
 | 
					    # TODO implement with `vagrant ssh` and rsync?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# TODO split out shared code into _podman.py, _vagrant.py, etc
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def push_wrapper(paths, appid, vercode, virt_container_type):
 | 
				
			||||||
 | 
					    """Push standard set of files into VM/container."""
 | 
				
			||||||
 | 
					    if virt_container_type == 'vagrant':
 | 
				
			||||||
 | 
					        vagrant_push(paths, appid, vercode)
 | 
				
			||||||
 | 
					    elif virt_container_type == 'podman':
 | 
				
			||||||
 | 
					        podman_push(paths, appid, vercode)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def make_file_list(appid, vercode):
 | 
				
			||||||
 | 
					    """Assemble list of files/folders that go into this specific build."""
 | 
				
			||||||
 | 
					    files = [f'build/{appid}', f'metadata/{appid}.yml']
 | 
				
			||||||
 | 
					    app_dir = f'metadata/{appid}'
 | 
				
			||||||
 | 
					    if Path(app_dir).exists():
 | 
				
			||||||
 | 
					        files.append(app_dir)
 | 
				
			||||||
 | 
					    app, build = metadata.get_single_build(appid, vercode)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    for lib in build.srclibs:
 | 
				
			||||||
 | 
					        srclib = common.getsrclib(lib, 'build/srclib', basepath=True, prepare=False)
 | 
				
			||||||
 | 
					        if srclib:
 | 
				
			||||||
 | 
					            # srclib metadata file
 | 
				
			||||||
 | 
					            files.append(f"srclibs/{srclib[0]}.yml")
 | 
				
			||||||
 | 
					            # srclib sourcecode
 | 
				
			||||||
 | 
					            files.append(srclib[2])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    extlib_paths = [f"build/extlib/{lib}" for lib in build.extlibs]
 | 
				
			||||||
 | 
					    missing_extlibs = [p for p in extlib_paths if not os.path.exists(p)]
 | 
				
			||||||
 | 
					    if any(missing_extlibs):
 | 
				
			||||||
 | 
					        raise Exception(
 | 
				
			||||||
 | 
					            "error: requested missing extlibs: {}".format(" ".join(missing_extlibs))
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					    files.extend(extlib_paths)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return files
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def create_build_dirs(appid, vercode, virt_container_type):
 | 
				
			||||||
 | 
					    """Create directories required for running builds."""
 | 
				
			||||||
 | 
					    dirs = ('build', 'build/extlib', 'build/srclib', 'metadata', 'fdroidserver')
 | 
				
			||||||
 | 
					    for dir_name in dirs:
 | 
				
			||||||
 | 
					        cmd = ['mkdir', '--parents', f'{common.BUILD_HOME}/{dir_name}']
 | 
				
			||||||
 | 
					        common.inside_exec(
 | 
				
			||||||
 | 
					            appid,
 | 
				
			||||||
 | 
					            vercode,
 | 
				
			||||||
 | 
					            cmd,
 | 
				
			||||||
 | 
					            virt_container_type,
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def full_push_sequence(appid, vercode, virt_container_type):
 | 
				
			||||||
 | 
					    """Push all files into vm required for specified build."""
 | 
				
			||||||
 | 
					    create_build_dirs(appid, vercode, virt_container_type)
 | 
				
			||||||
 | 
					    push_wrapper(make_file_list(appid, vercode), appid, vercode, virt_container_type)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # fdroidserver is pushed in every build
 | 
				
			||||||
 | 
					    cwd = Path('.').absolute()
 | 
				
			||||||
 | 
					    try:
 | 
				
			||||||
 | 
					        os.chdir(Path(__file__).parent.parent.parent)
 | 
				
			||||||
 | 
					        push_wrapper(
 | 
				
			||||||
 | 
					            [
 | 
				
			||||||
 | 
					                'fdroidserver/fdroidserver',
 | 
				
			||||||
 | 
					                'fdroidserver/fdroid',
 | 
				
			||||||
 | 
					            ],
 | 
				
			||||||
 | 
					            appid,
 | 
				
			||||||
 | 
					            vercode,
 | 
				
			||||||
 | 
					            virt_container_type,
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					    finally:
 | 
				
			||||||
 | 
					        os.chdir(cwd)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def main():
 | 
				
			||||||
 | 
					    """CLI main method for this subcommand."""
 | 
				
			||||||
 | 
					    parser = ArgumentParser(
 | 
				
			||||||
 | 
					        description="Push full build setup into the buildserver container/box."
 | 
				
			||||||
 | 
					    )
 | 
				
			||||||
 | 
					    common.setup_global_opts(parser)
 | 
				
			||||||
 | 
					    common.setup_virt_container_type_opts(parser)
 | 
				
			||||||
 | 
					    parser.add_argument(
 | 
				
			||||||
 | 
					        "APPID:VERCODE",
 | 
				
			||||||
 | 
					        help="Application ID with Version Code in the form APPID:VERCODE",
 | 
				
			||||||
 | 
					    )
 | 
				
			||||||
 | 
					    options = common.parse_args(parser)
 | 
				
			||||||
 | 
					    common.set_console_logging(options.verbose)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    try:
 | 
				
			||||||
 | 
					        appid, vercode = common.split_pkg_arg(options.__dict__['APPID:VERCODE'])
 | 
				
			||||||
 | 
					        full_push_sequence(
 | 
				
			||||||
 | 
					            appid,
 | 
				
			||||||
 | 
					            vercode,
 | 
				
			||||||
 | 
					            common.get_virt_container_type(options),
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					    except Exception as e:
 | 
				
			||||||
 | 
					        if options.verbose:
 | 
				
			||||||
 | 
					            logging.error(traceback.format_exc())
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            logging.error(e)
 | 
				
			||||||
 | 
					        sys.exit(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if __name__ == "__main__":
 | 
				
			||||||
 | 
					    main()
 | 
				
			||||||
							
								
								
									
										87
									
								
								tests/test_push.py
									
										
									
									
									
										Executable file
									
								
							
							
						
						
									
										87
									
								
								tests/test_push.py
									
										
									
									
									
										Executable file
									
								
							| 
						 | 
					@ -0,0 +1,87 @@
 | 
				
			||||||
 | 
					#!/usr/bin/env python3
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import importlib
 | 
				
			||||||
 | 
					import os
 | 
				
			||||||
 | 
					import unittest
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from pathlib import Path
 | 
				
			||||||
 | 
					from unittest import mock, skipIf
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from fdroidserver import common, exception, push
 | 
				
			||||||
 | 
					from .shared_test_code import mkdtemp, APPID, VERCODE, APPID_VERCODE
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class PushTest(unittest.TestCase):
 | 
				
			||||||
 | 
					    basedir = Path(__file__).resolve().parent
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def setUp(self):
 | 
				
			||||||
 | 
					        self._td = mkdtemp()
 | 
				
			||||||
 | 
					        self.testdir = self._td.name
 | 
				
			||||||
 | 
					        os.chdir(self.testdir)
 | 
				
			||||||
 | 
					        common.config = dict()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def tearDown(self):
 | 
				
			||||||
 | 
					        self._td.cleanup()
 | 
				
			||||||
 | 
					        common.config = None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class Push_main(PushTest):
 | 
				
			||||||
 | 
					    def setUp(self):
 | 
				
			||||||
 | 
					        super().setUp()
 | 
				
			||||||
 | 
					        metadatapath = Path(common.get_metadatapath(APPID))
 | 
				
			||||||
 | 
					        metadatapath.parent.mkdir()
 | 
				
			||||||
 | 
					        metadatapath.write_text(f'Name: Test\nBuilds:\n - versionCode: {VERCODE}\n')
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @mock.patch('sys.argv', ['fdroid push', APPID_VERCODE])
 | 
				
			||||||
 | 
					    @mock.patch('fdroidserver.push.create_build_dirs')
 | 
				
			||||||
 | 
					    @mock.patch('fdroidserver.push.podman_push')
 | 
				
			||||||
 | 
					    def test_podman(self, podman_push, create_build_dirs):
 | 
				
			||||||
 | 
					        common.config['virt_container_type'] = 'podman'
 | 
				
			||||||
 | 
					        push.main()
 | 
				
			||||||
 | 
					        create_build_dirs.assert_called_once()
 | 
				
			||||||
 | 
					        podman_push.assert_called()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @mock.patch('sys.argv', ['fdroid push', APPID_VERCODE])
 | 
				
			||||||
 | 
					    @mock.patch('fdroidserver.push.create_build_dirs')
 | 
				
			||||||
 | 
					    @mock.patch('fdroidserver.push.vagrant_push')
 | 
				
			||||||
 | 
					    def test_vagrant(self, vagrant_push, create_build_dirs):
 | 
				
			||||||
 | 
					        common.config['virt_container_type'] = 'vagrant'
 | 
				
			||||||
 | 
					        push.main()
 | 
				
			||||||
 | 
					        create_build_dirs.assert_called_once()
 | 
				
			||||||
 | 
					        vagrant_push.assert_called()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@skipIf(importlib.util.find_spec("podman") is None, 'Requires podman-py to run.')
 | 
				
			||||||
 | 
					class Push_podman_push(PushTest):
 | 
				
			||||||
 | 
					    def _only_run_if_container_exists(self):
 | 
				
			||||||
 | 
					        try:
 | 
				
			||||||
 | 
					            common.get_podman_container(APPID, VERCODE)
 | 
				
			||||||
 | 
					        except exception.BuildException as e:
 | 
				
			||||||
 | 
					            # To run these tests, first do: `./fdroid up com.example:123`
 | 
				
			||||||
 | 
					            self.skipTest(f'Requires Podman container {APPID_VERCODE} to run: {e}')
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def test_no_existing_container(self):
 | 
				
			||||||
 | 
					        appid = 'should.never.exist'
 | 
				
			||||||
 | 
					        f = Path(f'metadata/{appid}.yml')
 | 
				
			||||||
 | 
					        f.parent.mkdir()
 | 
				
			||||||
 | 
					        f.write_text(f.name)
 | 
				
			||||||
 | 
					        with self.assertRaises(exception.BuildException):
 | 
				
			||||||
 | 
					            push.podman_push(f, appid, 9999)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def test_bad_absolute_path(self):
 | 
				
			||||||
 | 
					        self._only_run_if_container_exists()
 | 
				
			||||||
 | 
					        with self.assertRaises(exception.BuildException):
 | 
				
			||||||
 | 
					            push.podman_push('/etc/passwd', APPID, VERCODE)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def test_bad_relative_path(self):
 | 
				
			||||||
 | 
					        self._only_run_if_container_exists()
 | 
				
			||||||
 | 
					        with self.assertRaises(ValueError):
 | 
				
			||||||
 | 
					            push.podman_push('../../etc/passwd', APPID, VERCODE)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def test_existing(self):
 | 
				
			||||||
 | 
					        self._only_run_if_container_exists()
 | 
				
			||||||
 | 
					        f = Path(f'metadata/{APPID}.yml')
 | 
				
			||||||
 | 
					        f.parent.mkdir()
 | 
				
			||||||
 | 
					        f.write_text(f.name)
 | 
				
			||||||
 | 
					        push.podman_push(f, APPID, VERCODE)
 | 
				
			||||||
 | 
					        common.inside_exec(APPID, VERCODE, ['test', '-e', str(f)], 'podman')
 | 
				
			||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue