# SPDX-FileCopyrightText: 2012, 2013, 2014, 2015, 2017, 2022, 2023 Michael Pyne # SPDX-FileCopyrightText: 2024 Andrew Shark # # SPDX-License-Identifier: GPL-2.0-or-later package ksb::BuildContext 0.35; use ksb; =head1 SYNOPSIS my $ctx = ksb::BuildContext->new(); $ctx->setRcFile('/path/to/kdesrc-buildrc'); my $fh = $ctx->loadRcFile(); ... for my $modName (@selectors) { $ctx->addModule(ksb::Module->new($modName, $ctx); } ... my @moduleList = $ctx->moduleList(); =head1 DESCRIPTION This contains the information needed about the build context, e.g. list of modules, what phases each module is in, the various options, etc. It also records information on which modules encountered errors (and what error), where to put log files, persistent options that should be available on the next run, and basically anything else that falls into the category of state management. =head2 The 'global' module One interesting thing about this class is that, as a state-managing class, this class implements the role of L for the pseudo-module called 'global' throughout the source code (and whose options are defined in the 'global' section in the rc-file). It is also a parent to every ksb::Module in terms of the option hierarchy, serving as a fallback source for ksb::Module's getOption() calls for most (though not all!) options. =cut use Carp 'confess'; use File::Basename; # dirname use IO::File; use POSIX qw(strftime); use Errno qw(:POSIX); use JSON::XS; # We derive from ksb::Module so that BuildContext acts like the 'global' # ksb::Module, with some extra functionality. # TODO: Derive from OptionsBase directly and remove getOption override use parent qw(ksb::Module); use ksb::BuildException; use ksb::Debug; use ksb::Util; use ksb::PhaseList; use ksb::Module; use ksb::Module::BranchGroupResolver; use ksb::Updater::KDEProjectMetadata 0.20; use ksb::Version qw(scriptVersion); use ksb::StatusView; use ksb::KDEProjectsReader 0.50; use File::Temp qw(tempfile); use File::Spec; # rel2abs use Mojo::File; # According to XDG spec, if $XDG_STATE_HOME is not set, then we should # default to ~/.local/state my $xdgStateHome = $ENV{XDG_STATE_HOME} // "$ENV{HOME}/.local/state"; my $xdgStateHomeShort = $xdgStateHome =~ s/^$ENV{HOME}/~/r; # Replace $HOME with ~ # According to XDG spec, if $XDG_CONFIG_HOME is not set, then we should # default to ~/.config my $xdgConfigHome = $ENV{XDG_CONFIG_HOME} // "$ENV{HOME}/.config"; my $xdgConfigHomeShort = $xdgConfigHome =~ s/^$ENV{HOME}/~/r; # Replace $HOME with ~ my @rcfiles = ("./kdesrc-buildrc", "$xdgConfigHome/kdesrc-buildrc", "$ENV{HOME}/.kdesrc-buildrc"); my $LOCKFILE_NAME = '.kdesrc-lock'; my $PERSISTENT_FILE_NAME = 'kdesrc-build-data'; my $SCRIPT_VERSION = scriptVersion(); # There doesn't seem to be a great way to get this from CMake easily but we can # reason that if there's a /usr/lib64 (and it's not just a compat symlink), # there will likely end up being a ${install-dir}/lib64 once kdesrc-build gets # done installing it my $libname = "lib"; $libname = "lib64" if (-d "/usr/lib64" and not -l "/usr/lib64"); $libname = "lib/x86_64-linux-gnu" if (-d "/usr/lib/x86_64-linux-gnu"); # These options are used for internal state, they are _not_ exposed as cmdline options my %GlobalOptions_private = ( "debug-level" => ksb::Debug::INFO, "filter-out-phases" => "", "git-desired-protocol" => "git", # protocol to use for git *push* URLs (fetch requires https) "git-repository-base" => { "qt6-copy" => "https://invent.kde.org/qt/qt/", "_" => "fake/" }, "manual-build" => "", "manual-update" => "", "module-definitions-dir" => ($ENV{XDG_STATE_HOME} // "$ENV{HOME}/.local/state") . "/sysadmin-repo-metadata/module-definitions", "repository" => "", # module's git repo "set-env" => {}, # Hash of environment vars to set "ssh-identity-file" => "", # If set, is passed to ssh-add. "use-modules" => "", ); # These options are exposed as cmdline options, but _not from here_. # Their more complex specifier is made in ksb::Cmdline _supportedOptions(). my %GlobalOptions_with_extra_specifier = ( "build-when-unchanged" => 1, "colorful-output" => 1, "ignore-modules" => "", "niceness" => "10", # todo convert to int? "pretend" => "", "refresh-build" => "", ); # These options are exposed as cmdline options without parameters, and having the negatable form with "--no-". our %GlobalOptions_with_negatable_form = ( "async" => 1, "compile-commands-export" => 1, # 2021-02-06 allow to generate compile_commands.json via cmake, for clangd tooling "compile-commands-linking" => 0, # 2021-02-06 link generated compile_commands.json back to the source directory "delete-my-patches" => 0, # Should only be set from cmdline "delete-my-settings" => 0, # Should only be set from cmdline "disable-agent-check" => 0, # If true we don't check on ssh-agent "generate-vscode-project-config" => 0, "include-dependencies" => 1, "install-after-build" => 1, "install-environment-driver" => 1, # Setup ~/.config/kde-env-*.sh for login scripts "install-session-driver" => 0, # Above, + ~/.xsession "purge-old-logs" => 1, "run-tests" => 0, # 1 = make test, upload = make Experimental # todo why boolean option may have "upload" value? "stop-on-failure" => 1, "use-clean-install" => 0, "use-idle-io-priority" => 0, "use-inactive-modules" => 0, ); # These options are exposed as cmdline options that require some parameter our %GlobalOptions_with_parameter = ( "binpath" => "", "branch" => "", "branch-group" => "", # Overrides branch, uses JSON data. "build-dir" => "$ENV{HOME}/kde/build", "cmake-generator" => "", "cmake-options" => "", "cmake-toolchain" => "", "configure-flags" => "", "custom-build-command" => "", "cxxflags" => "-pipe", "directory-layout" => "flat", "dest-dir" => '${MODULE}', # single quotes used on purpose! "do-not-compile" => "", "http-proxy" => "", # Proxy server to use for HTTP. "install-dir" => "$ENV{HOME}/kde/usr", "libname" => $libname, "libpath" => "", "log-dir" => "log", "make-install-prefix" => "", # Some people need sudo "make-options" => "", "ninja-options" => "", "num-cores" => "", # Used for build constraints "num-cores-low-mem" => 2, # Used only in rc-file but documented "override-build-system" => "", "persistent-data-file" => "", "qmake-options" => "", "qt-install-dir" => "", "remove-after-install" => "none", # { none, builddir, all } "revision" => "", # Was only useful for Subversion modules at cmdline. TODO Implement for git? "source-dir" => "$ENV{HOME}/kde/src", "tag" => "", ); # These options are exposed as cmdline options without parameters our %GlobalOptions_without_parameter = ( "build-system-only" => "", "reconfigure" => "", "metadata-only" => "", ); sub new ($class) { # It is very important to use the ksb::Module:: syntax instead of ksb::Module->, # otherwise you can't pass $class and have it used as the classname. my $self = ksb::Module::new($class, undef, 'global'); my %newOpts = ( modules => [], context => $self, # Fix link to buildContext (i.e. $self) build_options => { global => { %GlobalOptions_private, %GlobalOptions_with_extra_specifier, %GlobalOptions_without_parameter, %GlobalOptions_with_negatable_form, %GlobalOptions_with_parameter, }, # Module options are stored under here as well, keyed by module->name() }, # This one replaces ksb::Module::{phases} phases => ksb::PhaseList->new(), errors => { # A map from module *names* (as in modules[] above) to the # phase name at which they failed. }, logPaths=> { # Holds a hash table of log path bases as expanded by # getSubdirPath (e.g. [source-dir]/log) to the actual log dir # *this run*, with the date and unique id added. You must still # add the module name to use. }, rcFiles => [@rcfiles], rcFile => undef, env => { }, persistent_options => { }, # These are kept across multiple script runs ignore_list => [ ], # List of KDE project paths to ignore completely kde_projects_metadata => undef, # Enumeration of kde-projects logical_module_resolver => undef, # For branch-group option status_view => ksb::StatusView->new(), projects_db => undef, # See getProjectDataReader ); # Merge all new options into our self-hash. @{$self}{keys %newOpts} = values %newOpts; $self->{options} = $self->{build_options}{global}; assert_isa($self, 'ksb::Module'); assert_isa($self, 'ksb::BuildContext'); return $self; } # Gets the ksb::PhaseList for this context, and optionally sets it first to # the ksb::PhaseList passed in. sub phases { my ($self, $phases) = @_; if ($phases) { confess("Invalid type, expected PhaseList") unless $phases->isa('ksb::PhaseList'); $self->{phases} = $phases; } return $self->{phases}; } sub addModule { my ($self, $module) = @_; Carp::confess("No module to push") unless $module; my $path; if (list_has($self->{modules}, $module)) { debug("Skipping duplicate module ", $module->name()); } elsif (($path = $module->fullProjectPath()) && # See if the name matches any given in the ignore list. any(sub { $path =~ /(^|\/)$_($|\/)/ }, $self->{ignore_list})) { debug("Skipping ignored module $module"); } else { debug("Adding $module to module list"); push @{$self->{modules}}, $module; } } # Returns a listref of the modules to build sub moduleList { my $self = shift; return $self->{modules}; } # Adds a list of modules to ignore processing on completely. # Parameters should simply be a list of KDE project paths to ignore, # e.g. 'extragear/utils/kdesrc-build'. Partial paths are acceptable, matches # are determined by comparing the path provided to the suffix of the full path # of modules being compared. See KDEProjectsReader::_projectPathMatchesWildcardSearch # # Existing items on the ignore list are not removed. sub addToIgnoreList { my $self = shift; push @{$self->{ignore_list}}, @_; } sub setupOperatingEnvironment { my $self = shift; # Set the process priority POSIX::nice(int $self->getOption('niceness')); # Set the IO priority if available. if ($self->getOption('use-idle-io-priority')) { # -p $$ is our PID, -c3 is idle priority # 0 return value means success if (safe_system('ionice', '-c3', '-p', $$) != 0) { warning (" b[y[*] Unable to lower I/O priority, continuing..."); } } # Get ready for logged output. ksb::Debug::setLogFile($self->getLogDirFor($self) . '/build-log'); # Propagate HTTP proxy through environment unless overridden. if ((my $proxy = $self->getOption('http-proxy')) && !defined $ENV{'http_proxy'}) { $self->queueEnvironmentVariable('http_proxy', $proxy); } } # Clears the list of environment variables to set for log_command runs. sub resetEnvironment { my $self = assert_isa(shift, 'ksb::BuildContext'); $self->{env} = { }; } # Adds an environment variable and value to the list of environment # variables to apply for the next subprocess execution. # # Note that these changes are /not/ reflected in the current environment, # so if you are doing something that requires that kind of update you # should do that yourself (but remember to have some way to restore the old # value if necessary). # # In order to keep compatibility with the old 'setenv' sub, no action is # taken if the value is not equivalent to boolean true. sub queueEnvironmentVariable { my $self = assert_isa(shift, 'ksb::BuildContext'); my ($key, $value) = @_; return unless $value; debug ("\tQueueing g[$key] to be set to y[$value]"); $self->{env}->{$key} = $value; } # Applies all changes queued by queueEnvironmentVariable to the actual # environment irretrievably. Use this before exec()'ing another child, for # instance. sub commitEnvironmentChanges { my $self = assert_isa(shift, 'ksb::BuildContext'); while (my ($key, $value) = each %{$self->{env}}) { $ENV{$key} = $value; debug ("\tSetting environment variable g[$key] to g[b[$value]"); } } # Adds the given library paths to the path already given in an environment # variable. In addition, detected "system paths" are stripped to ensure # that we don't inadvertently re-add a system path to be promoted over the # custom code we're compiling (for instance, when a system Qt is used and # installed to /usr). # # If the environment variable to be modified has already been queued using # queueEnvironmentVariable, then that (queued) value will be modified and # will take effect with the next forked subprocess. # # Otherwise, the current environment variable value will be used, and then # queued. Either way the current environment will be unmodified afterward. # # First parameter is the name of the environment variable to modify # All remaining parameters are prepended to the current environment path, in # the order given. (i.e. param1, param2, param3 -> # param1:param2:param3:existing) sub prependEnvironmentValue ($self, $envName, @items) { my @curPaths = split(':', $self->{env}->{$envName} // $ENV{$envName} // ''); # Filter out entries to add that are already in the environment from # the system. for my $path (grep { list_has(\@curPaths, $_) } (@items) ) { debug ("\tNot prepending y[$path] to y[$envName] as it appears " . "to already be defined in y[$envName]."); } @items = grep { not list_has(\@curPaths, $_); } (@items); my $envValue = join(':', @items, @curPaths); $envValue =~ s/^:*//; $envValue =~ s/:*$//; # Remove leading/trailing colons $envValue =~ s/:+/:/; # Remove duplicate colons $self->queueEnvironmentVariable($envName, $envValue); } # Tries to take the lock for our current base directory, which currently is # what passes for preventing people from accidentally running kdesrc-build # multiple times at once. The lock is based on the base directory instead # of being global to allow for motivated and/or brave users to properly # configure kdesrc-build to run simultaneously with different # configurations. # # Return value is a boolean success flag. sub takeLock { my $self = assert_isa(shift, 'ksb::BuildContext'); my $baseDir = $self->baseConfigDirectory(); my $lockfile = "$baseDir/$LOCKFILE_NAME"; $! = 0; # Force reset to non-error status sysopen LOCKFILE, $lockfile, O_WRONLY | O_CREAT | O_EXCL; my $errorCode = $!; # Save for later testing. if ($errorCode == EEXIST) { # Path already exists, read the PID and see if it belongs to a # running process. open (my $pidFile, "<", $lockfile) or do { # Lockfile is there but we can't open it?!? Maybe a race # condition but I have to give up somewhere. warning (" WARNING: Can't open or create lockfile r[$lockfile]"); return 1; }; my $pid = <$pidFile>; close $pidFile; if ($pid) { # Recent kdesrc-build; we wrote a PID in there. chomp $pid; # See if something's running with this PID. if (kill(0, $pid) == 1) { # Something *is* running, likely kdesrc-build. Don't use error, # it'll scan for $! print ksb::Debug::colorize(" r[*y[*r[*] kdesrc-build appears to be running. Do you want to:\n"); print ksb::Debug::colorize(" (b[Q])uit, (b[P])roceed anyways?: "); my $choice = // ''; chomp $choice; if (lc $choice ne 'p') { say ksb::Debug::colorize(" y[*] kdesrc-build run canceled."); return 0; } # We still can't grab the lockfile, let's just hope things # work out. note (" y[*] kdesrc-build run in progress by user request."); return 1; } # If we get here, then the program isn't running (or at least not # as the current user), so allow the flow of execution to fall # through below and unlink the lockfile. } # pid # No pid found, optimistically assume the user isn't running # twice. warning (" y[WARNING]: stale kdesrc-build lockfile found, deleting."); unlink $lockfile; sysopen (LOCKFILE, $lockfile, O_WRONLY | O_CREAT | O_EXCL) or do { error (" r[*] Still unable to lock $lockfile, proceeding anyways..."); return 1; }; # Hope the sysopen worked... fall-through } elsif ($errorCode == ENOTTY) { # Stupid bugs... normally sysopen will return ENOTTY, not sure who's to blame between # glibc and Perl but I know that setting PERLIO=:stdio in the environment "fixes" things. ; # pass } elsif ($errorCode != 0) # Some other error occurred. { warning (" r[*]: Error $errorCode while creating lock file (is $baseDir available?)"); warning (" r[*]: Continuing the script for now..."); # Even if we fail it's generally better to allow the script to proceed # without being a jerk about things, especially as more non-CLI-skilled # users start using kdesrc-build to build KDE. return 1; } say LOCKFILE "$$"; close LOCKFILE; return 1; } # Releases the lock obtained by takeLock. sub closeLock { my $self = assert_isa(shift, 'ksb::BuildContext'); my $baseDir = $self->baseConfigDirectory(); my $lockFile = "$baseDir/$LOCKFILE_NAME"; unlink ($lockFile) or warning(" y[*] Failed to close lock: $!"); } # This subroutine accepts a Module parameter, and returns the log directory # for it. You can also pass a BuildContext (including this one) to get the # default log directory. # # As part of setting up what path to use for the log directory, the # 'latest' symlink will also be setup to point to the returned log # directory. sub getLogDirFor { my ($self, $module) = @_; my $baseLogPath = $module->getSubdirPath('log-dir'); my $logDir; if (!exists $self->{logPaths}{$baseLogPath}) { # No log dir made for this base, do so now. my $id = '01'; my $date = strftime "%F", localtime; # ISO 8601 date $id++ while -e "$baseLogPath/$date-$id"; $self->{logPaths}{$baseLogPath} = "$baseLogPath/$date-$id"; } $logDir = $self->{logPaths}{$baseLogPath}; super_mkdir($logDir); # global logs go to basedir directly $logDir .= "/$module" unless $module->isa('ksb::BuildContext'); return $logDir; } # Constructs the appropriate full path to a log file based on the given # basename (including extensions). Use this instead of getLogDirFor when you # actually intend to create a log, as this function will also adjust the # 'latest' symlink properly. sub getLogPathFor { my ($self, $module, $path) = @_; my $baseLogPath = $module->getSubdirPath('log-dir'); my $logDir = $self->getLogDirFor($module); # We create this here to avoid needless empty module directories everywhere super_mkdir($logDir); # Provide a directory to make it easy to see the last build for a module's # given phase (like cmake, build, install, etc.) without having to find the # log dir for the specific kdesrc-build run. super_mkdir("$baseLogPath/latest-by-phase/$module"); # Add a symlink to the latest run for this module. 'latest' itself is # a directory under the base log directory that holds symlinks mapping # each module name to the specific log directory most recently used. my $latestPath = "$baseLogPath/latest"; # Handle stuff like playground/utils or KDE/kdelibs my ($moduleName, $modulePath) = fileparse($module->name()); $latestPath .= "/$modulePath" if $module->name() =~ m(/); super_mkdir($latestPath); my $symlink = "$latestPath/$moduleName"; remake_symlink($logDir, $symlink); my $symlink2 = "$baseLogPath/latest-by-phase/$module/$path"; remake_symlink("$logDir/$path", $symlink2); return "$logDir/$path"; } # Returns rc file in use. Call loadRcFile first. sub rcFile { my $self = shift; return $self->{rcFile}; } # Forces the rc file to be read from to be that given by the first # parameter. sub setRcFile { my ($self, $file) = @_; $self->{rcFiles} = [$file]; $self->{rcFile} = undef; } # Warns a user if the config file is stored in the old location. sub warnLegacyConfig { my $file = shift; $file =~ s/^$ENV{HOME}/~/; if ($file eq '~/.kdesrc-buildrc') { warning (<{rcFiles}}; my $fh; for my $file (@rcFiles) { if (open ($fh, '<', "$file")) { $self->{rcFile} = File::Spec->rel2abs($file); warnLegacyConfig($file); return $fh; } } # No rc found, check if we can use default. if (scalar @rcFiles == 1) { # This can only happen if the user uses --rc-file, so if we fail to # load the file, we need to fail to load at all. my $failedFile = $rcFiles[0]; error (<<~EOM); Unable to open config file $failedFile Script stopping here since you specified --rc-file on the command line to load $failedFile manually. If you wish to run the script with no configuration file, leave the --rc-file option out of the command line. If you want to force an empty rc file, use --rc-file /dev/null EOM croak_runtime("Missing $failedFile"); } if ($self->getOption("metadata-only")){ # If configuration file in default location was not found, and no --rc-file option was used, and metadata-only option was used. # In FirstRun user may decide to use --install-distro-packages before --generate-config. # --install-distro-packages requires the metadata to be downloaded to read the "disrto-dependencies" from it. # After downloading metadata, we normally should change 'last-metadata-update' persistent option value. # To store persistent option, we should know persistent-data-file value, and it is read from config. # At this moment we know that there is no config at default location, and user did not specified the --rc-file option. # And because we do not want to _require_ the config to be available yet, we just will provide dummy config. # This way the --metadata-only option could work in both cases: when user has config and when he has not. # When he has config (not current case), the persistent option "last-metadata-update" will be set as expected, and after the build process will be stored in persistent file. # When he has no config (the current case), we will let the _readConfigurationOptions function do its work on fake config, then we will return. my $dummyConfig = <<~END; global persistent-data-file /not/existing/file # should not exist in file system (so it is not tried to be read, otherwise we should provide a valid json) end global # To suppress warning about no modules in configuration. module fake end module END open $fh, '<', \$dummyConfig; $self->{rcFile} = "/fake/dummy_config"; return $fh; } else { # If no configuration and no --rc-file option was used, warn the user and fail. error (<<~EOM); b[No configuration file is present.] kdesrc-build requires a configuration file to select which KDE software modules to build, what options to build them with, the path to install to, etc. When run, kdesrc-build will use `kdesrc-buildrc' config file located in the current working directory. If no such file exists, kdesrc-build will use `$xdgConfigHomeShort/kdesrc-buildrc' instead. You can generate config with b[--generate-config]. EOM croak_runtime("No configuration available"); } } # Returns the base directory that holds the configuration file. This is # typically used as the directory base for other necessary kdesrc-build # execution files, such as the persistent data store and lock file. # # The RC file must have been found and loaded first, obviously. sub baseConfigDirectory { my $self = assert_isa(shift, 'ksb::BuildContext'); my $rcfile = $self->rcFile() or croak_internal("Call to baseConfigDirectory before loadRcFile"); return dirname($rcfile); } sub modulesInPhase ($self, $phase) { my @list = grep { $_->phases()->has($phase) } (@{$self->moduleList()}); return @list; } sub usesConcurrentPhases ($self) { # If we have an 'update' phase and any other phase (build / test / install # / etc) we should use concurrency if it is available. my $has_update = 0; my $has_other = 0; for my $mod (@{$self->moduleList()}) { for my $phase ($mod->phases()->phases()) { if ($phase eq 'update') { $has_update = 1; } else { $has_other = 1; } } return 1 if ($has_update && $has_other); } return 0; } # Searches for a module with a name that matches the provided parameter, # and returns its ksb::Module object. Returns undef if no match was found. # As a special-case, returns the BuildContext itself if the name passed is # 'global', since the BuildContext also is a (in the "is-a" OOP sense) # ksb::Module, specifically the 'global' one. sub lookupModule { my ($self, $moduleName) = @_; return $self if $moduleName eq 'global'; my @options = grep { $_->name() eq $moduleName } (@{$self->moduleList()}); return undef unless @options; if (scalar @options > 1) { croak_internal("Detected 2 or more $moduleName ksb::Module objects"); } return $options[0]; } sub markModulePhaseFailed { my ($self, $phase, $module) = @_; assert_isa($module, 'ksb::Module'); $self->{errors}->{$module->name()} = $phase; } # Returns a list (i.e. not a reference to, but a real list) of Modules that failed to # complete the given phase. sub failedModulesInPhase { my ($self, $phase) = @_; my @failures = grep { ($self->{errors}->{$_->name()} // '') eq $phase } (@{$self->moduleList()}); return @failures; } # Returns a list of modules that had a failure of some sort, in the order the modules # are listed in our current module list. sub listFailedModules { my $self = shift; my @modules = @{$self->moduleList()}; # grepping for failures instead of returning error list directly maintains ordering @modules = grep { exists $self->{errors}->{$_->name()} } (@modules); return @modules; } # OVERRIDE: Our immediate parent class Module overrides this, but we actually # want the OptionsBase version to be used instead, until we break the recursive # use of Module's own getOption calls on our getOption. sub getOption ($self, $key) { $self->ksb::OptionsBase::getOption($key); } # OVERRIDE sub setOption { my ($self, %options) = @_; # Special case handling. if (exists $options{'filter-out-phases'}) { for my $phase (split(' ', $options{'filter-out-phases'})) { $self->phases()->filterOutPhase($phase); } delete $options{'filter-out-phases'}; } # Our immediate parent class Module overrides this, but we actually # want the OptionsBase version to be used instead, because Module's version specifically checks for # some options prohibited for it (such as "ignore-modules") but we may want such for BuildContext. $self->ksb::OptionsBase::setOption(%options); # Automatically respond to various global option changes. while (my ($key, $value) = each %options) { my $normalizedKey = $key; $normalizedKey =~ s/^#//; # Remove sticky key modifier. if ($normalizedKey eq 'colorful-output') { ksb::Debug::setColorfulOutput($value); } elsif ($normalizedKey eq 'debug-level') { ksb::Debug::setDebugLevel($value); } elsif ($normalizedKey eq 'pretend') { ksb::Debug::setPretending($value); } } } # # Persistent option handling # # Returns the name of the file to use for persistent data. sub persistentOptionFileName { my $self = shift; my $file = $self->getOption('persistent-data-file'); if ($file) { $file =~ s/^~/$ENV{HOME}/; } else { my $configDir = $self->baseConfigDirectory(); if ($configDir eq $xdgConfigHome) { # Global config is used. Store the data file in XDG_STATE_HOME. $file = $xdgStateHome . '/' . $PERSISTENT_FILE_NAME; } else { # Local config is used. Store the data file in the same directory. $file = $configDir . '/.' . $PERSISTENT_FILE_NAME; } my @rcFiles = @{$self->{rcFiles}}; if (scalar @rcFiles == 1) { # This can only mean that the user specified an rcfile on the command # line and did not set persistent-data-file in their config file. In # this case, append the name of the rcfile to the persistent build # data file to associate it with that specific rcfile. my $rcFilePath = $rcFiles[0]; # ...But only if the specified rcfile isn't one of the default ones, # to prevent the user from making an oopsie if (grep { $_ eq $rcFilePath } @rcfiles) { warning("The specified rc file is one of the default ones. Ignoring it."); } else { my $rcFileName = basename($rcFilePath); $file = "${file}-$rcFileName"; } } # Fallback to legacy data file if it exists and the new one doesn't. my $legacyDataFile = "$ENV{HOME}/.kdesrc-build-data"; if (! -e $file && -e $legacyDataFile) { $file = $legacyDataFile; } if ($file eq $legacyDataFile && !$self->getOption('#warned-legacy-data-location')) { warning (<setOption('#warned-legacy-data-location', 1); } } return $file; } # Reads in all persistent options from the file where they are kept # (kdesrc-build-data) for use in the program. # # The directory used is the same directory that contains the rc file in use. sub loadPersistentOptions ($self) { # We need to keep persistent data with the context instead of with the # applicable modules since otherwise we might forget to write out # persistent data for modules we didn't build in this run. So, we just # store it all. # # Layout of this data: # $self->persistent_options = { # 'module-name' => { # option => value, # # foreach option/value pair # }, # # foreach module # } $self->{persistent_options} = {}; my $fname = $self->persistentOptionFileName(); return unless -e $fname; my $persistent_data = Mojo::File->new($fname)->slurp; # $persistent_data should be a JSON object which we can store directly as a # hash. my $persistent_options = eval { decode_json($persistent_data) }; if ($@ || (ref $persistent_options ne 'HASH')) { error ("Failed to read persistent module data: r[b[$@]"); return; } $self->{persistent_options} = $persistent_options; } # Writes out persistent options to the kdesrc-build-data file. # # The directory used is the same directory that contains the rc file in use. sub storePersistentOptions ($self) { return if pretending(); my $fileName = $self->persistentOptionFileName(); my $dir = dirname($fileName); super_mkdir($dir) unless -d $dir; eval { my $encodedJSON = JSON::XS->new->pretty(1)->canonical(1)->encode($self->{persistent_options}); Mojo::File->new($fileName)->spew($encodedJSON); }; if ($@) { error ("Unable to save persistent module data: b[r[$!]"); return; } } # Returns the value of a "persistent" option (normally read in as part of # startup), or undef if there is no value stored. # # First parameter is the module name to get the option for, or 'global' if # not for a module. # Note that unlike setOption/getOption, no inheritance is done at this # point so if an option is present globally but not for a module you # must check both if that's what you want. # Second parameter is the name of the value to retrieve (i.e. the key) sub getPersistentOption { my ($self, $moduleName, $key) = @_; my $persistent_opts = $self->{persistent_options}; # We must check at each level of indirection to avoid # "autovivification" return unless exists $persistent_opts->{$moduleName}; return unless exists $persistent_opts->{$moduleName}{$key}; return $persistent_opts->{$moduleName}{$key}; } # Clears a persistent option if set (for a given module and option-name). # # First parameter is the module name to get the option for, or 'global' for # the global options. # Second parameter is the name of the value to clear. # No return value. sub unsetPersistentOption { my ($self, $moduleName, $key) = @_; my $persistent_opts = $self->{persistent_options}; if (exists $persistent_opts->{$moduleName} && exists $persistent_opts->{$moduleName}->{$key}) { delete $persistent_opts->{$moduleName}->{$key}; } } # Sets a "persistent" option which will be read in for a module when # kdesrc-build starts up and written back out at (normal) program exit. # # First parameter is the module name to set the option for, or 'global'. # Second parameter is the name of the value to set (i.e. key) # Third parameter is the value to store, which must be a scalar. sub setPersistentOption { my ($self, $moduleName, $key, $value) = @_; my $persistent_opts = $self->{persistent_options}; # Initialize empty hash ref if nothing defined for this module. $persistent_opts->{$moduleName} //= { }; $persistent_opts->{$moduleName}{$key} = $value; } # Returns the ksb::Module (which has a 'metadata' scm type) that is used for # kde-project metadata, so that other modules that need it can call into it if # necessary. # # Also may return undef if the metadata is unavailable or has not yet # been set by setKDEProjectsMetadataModule (this method does not # automatically create the needed module). sub getKDEProjectsMetadataModule { my $self = shift; # Initialize if not set $self->{kde_projects_metadata} //= ksb::ModuleSet::KDEProjects::getProjectMetadataModule($self); return $self->{kde_projects_metadata}; } # Returns a KDEProjectsReader module, which has already read in the database and # is ready to be queried. Note that exceptions can be thrown in the process # of downloading and parsing the database information, so be ready for that. sub getProjectDataReader { my $self = shift; return $self->{projects_db} if $self->{projects_db}; my $projectDatabaseModule = $self->getKDEProjectsMetadataModule() or croak_runtime("kde-projects repository information could not be downloaded: $!"); $self->{projects_db} = ksb::KDEProjectsReader->new($projectDatabaseModule); return $self->{projects_db}; } # Returns the effective branch group to use for modules. You should not call # this unless KDE project metadata is available (see # setKDEProjectsMetadataModule and moduleBranchGroupResolver). sub effectiveBranchGroup { my $self = shift; my $branchGroup = $self->getOption('branch-group') // 'kf5-qt5'; return $branchGroup; } # Returns a ksb::Module::BranchGroupResolver which can be used to efficiently # determine a git branch to use for a given kde-projects module (when the # branch-group option is in use), as specified at # https://community.kde.org/Infrastructure/Project_Metadata. sub moduleBranchGroupResolver { my $self = shift; if (!$self->{logical_module_resolver}) { my $metadataModule = $self->getKDEProjectsMetadataModule(); croak_internal("Tried to use branch-group, but needed data wasn't loaded!") unless $metadataModule; my $resolver = ksb::Module::BranchGroupResolver->new( $metadataModule->scm()->logicalModuleGroups()); $self->{logical_module_resolver} = $resolver; } return $self->{logical_module_resolver}; } sub statusViewer { my $self = shift; return $self->{status_view}; } 1;