123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107 |
- #
- # Copyright OpenEmbedded Contributors
- #
- # SPDX-License-Identifier: GPL-2.0-only
- #
- import logging
- import os
- import stat
- import sys
- import shutil
- import bb.utils
- import bb.process
- from bblayers.common import LayerPlugin
- logger = logging.getLogger('bitbake-layers')
- sys.path.insert(0, os.path.dirname(os.path.dirname(__file__)))
- import oe.buildcfg
- def plugin_init(plugins):
- return MakeSetupPlugin()
- class MakeSetupPlugin(LayerPlugin):
- def _get_repo_path(self, layer_path):
- repo_path, _ = bb.process.run('git rev-parse --show-toplevel', cwd=layer_path)
- return repo_path.strip()
- def _get_remotes(self, repo_path):
- remotes = {}
- remotes_list,_ = bb.process.run('git remote', cwd=repo_path)
- for r in remotes_list.split():
- uri,_ = bb.process.run('git remote get-url {r}'.format(r=r), cwd=repo_path)
- remotes[r] = {'uri':uri.strip()}
- return remotes
- def _get_describe(self, repo_path):
- try:
- describe,_ = bb.process.run('git describe --tags', cwd=repo_path)
- except bb.process.ExecutionError:
- return ""
- return describe.strip()
- def make_repo_config(self, destdir):
- """ This is a helper function for the writer plugins that discovers currently confugured layers.
- The writers do not have to use it, but it can save a bit of work and avoid duplicated code, hence it is
- available here. """
- repos = {}
- layers = oe.buildcfg.get_layer_revisions(self.tinfoil.config_data)
- try:
- destdir_repo = self._get_repo_path(destdir)
- except bb.process.ExecutionError:
- destdir_repo = None
- for (l_path, l_name, l_branch, l_rev, l_ismodified) in layers:
- if l_name == 'workspace':
- continue
- if l_ismodified:
- logger.error("Layer {name} in {path} has uncommitted modifications or is not in a git repository.".format(name=l_name,path=l_path))
- return
- repo_path = self._get_repo_path(l_path)
- if repo_path not in repos.keys():
- repos[repo_path] = {'path':os.path.basename(repo_path),'git-remote':{'rev':l_rev, 'branch':l_branch, 'remotes':self._get_remotes(repo_path), 'describe':self._get_describe(repo_path)}}
- if repo_path == destdir_repo:
- repos[repo_path]['contains_this_file'] = True
- if not repos[repo_path]['git-remote']['remotes'] and not repos[repo_path]['contains_this_file']:
- logger.error("Layer repository in {path} does not have any remotes configured. Please add at least one with 'git remote add'.".format(path=repo_path))
- return
- top_path = os.path.commonpath([os.path.dirname(r) for r in repos.keys()])
- repos_nopaths = {}
- for r in repos.keys():
- r_nopath = os.path.basename(r)
- repos_nopaths[r_nopath] = repos[r]
- r_relpath = os.path.relpath(r, top_path)
- repos_nopaths[r_nopath]['path'] = r_relpath
- return repos_nopaths
- def do_make_setup(self, args):
- """ Writes out a configuration file and/or a script that replicate the directory structure and revisions of the layers in a current build. """
- for p in self.plugins:
- if str(p) == args.writer:
- p.do_write(self, args)
- def register_commands(self, sp):
- parser_setup_layers = self.add_command(sp, 'create-layers-setup', self.do_make_setup, parserecipes=False)
- parser_setup_layers.add_argument('destdir',
- help='Directory where to write the output\n(if it is inside one of the layers, the layer becomes a bootstrap repository and thus will be excluded from fetching).')
- parser_setup_layers.add_argument('--output-prefix', '-o',
- help='File name prefix for the output files, if the default (setup-layers) is undesirable.')
- self.plugins = []
- for path in (self.tinfoil.config_data.getVar('BBPATH').split(':')):
- pluginpath = os.path.join(path, 'lib', 'bblayers', 'setupwriters')
- bb.utils.load_plugins(logger, self.plugins, pluginpath)
- parser_setup_layers.add_argument('--writer', '-w', choices=[str(p) for p in self.plugins], help="Choose the output format (defaults to oe-setup-layers).\n\nCurrently supported options are:\noe-setup-layers - a self-contained python script and a json config for it.\n\n", default="oe-setup-layers")
- for plugin in self.plugins:
- if hasattr(plugin, 'register_arguments'):
- plugin.register_arguments(parser_setup_layers)
|