#!/usr/bin/perl use Encode 'encode'; use FindBin; use Mojolicious; use Mojo::File 'path'; use Mojo::Log; use Mojo::Util; use warnings; use strict; use utf8; my @vcs_0_variant_suffixes = (qw(git svn hg)); my @cfg_0_variant_suffixes = (qw(static static-compat)); my @cfg_1_variant_suffixes = (qw(doc test cli angle dynamic opengl noopengl)); my @variant_suffixes = (\@vcs_0_variant_suffixes, \@cfg_0_variant_suffixes, \@cfg_1_variant_suffixes); sub is_outdated { my ($source_path, $target_path) = @_; my $source_last_modified = (stat($source_path))[9]; my $target_last_modified = (stat($target_path))[9]; return !defined $target_last_modified || $source_last_modified > $target_last_modified; } my $log = Mojo::Log->new; my $mojolicious = Mojolicious->new; my $renderer = $mojolicious->renderer; my $pkgbuilds_dir = path($FindBin::Bin, '..', '..')->realpath; my @template_paths = ("$FindBin::Bin/templates", $pkgbuilds_dir); $log->level($ENV{LOG_LEVEL} // 'info'); $mojolicious->log($log); $renderer->paths(\@template_paths); $log->debug("Template paths:\n" . join("\n", @template_paths)); my $filter_regex = $ARGV[0]; my $install_directory = $ARGV[1] // $pkgbuilds_dir; unless (-d $install_directory) { $log->error("Output directory '$install_directory' does not exist."); exit(-1); } # add helper to render Qt dependencies sub _render_deps { my ($package_prefix, $controller, @d) = @_; my $prefix = $controller->stash('package_name_prefix'); my $suffix = $controller->stash('package_name_suffix'); my $quote = $prefix =~ qr/^(mingw-w64|android)/ ? "'" : ''; return join(' ', map { "${quote}${prefix}${package_prefix}-${_}${suffix}${quote}" } @d); } sub _render_optdeps { my ($package_prefix, $controller, %d) = @_; my $prefix = $controller->stash('package_name_prefix'); my $suffix = $controller->stash('package_name_suffix'); return join(' ', map { "'${prefix}${package_prefix}-${_}${suffix}: $d{$_}'" } sort keys %d); } for my $qt_version (qw(qt5 qt6)) { $mojolicious->helper("${qt_version}deps" => sub { _render_deps($qt_version, @_) }); $mojolicious->helper("${qt_version}optdeps" => sub { _render_optdeps($qt_version, @_) }); } # add helper to expand pkg-config style libraries into full paths for use with CMake # example: "-lfoo -lbar" => "/usr/lib/foo.a;/usr/lib/bar.a" $mojolicious->helper(expand_libs => sub { my $controller = shift; my $prefix = $controller->stash('install_prefix'); my $extension = $controller->stash('library_extension'); return join(';', map { my $library_name = $_; $library_name = $1 if $library_name =~ qr/\w*-l(.*)\w*/; "$prefix/lib/lib$library_name.$extension"; } @_); }); # define revisions of Qt modules from KDE fork my %kde_fork_revisions = ( # module => [rev, 'commit on KDE fork', 'version bump to be reverted'], activeqt => [0, '2334cafc110c4e63bec3a5c7abdcd67e5e5ee754'], winextras => [0, '48318520a4031167c4c0ad559e1a11b2f4c053d6'], '3d' => [0, 'c3c7e6ebc29cce466d954f72f340a257d76b5ec2'], base => [157, '9cf586d629a04040c9414c4f9e17abbc65c644eb'], charts => [0, 'e30be213e483f2d6f3c40af0cbdc11a8e92e2026'], connectivity => [5, '056294c0493f814c3951ef57e5d0656efe643fb5'], datavis3d => [0, '9f0f50ebef04d5ac5ed0ee2a3a71e7748fce3005'], declarative => [21, '8defe7bfcae2ac5cb6dc25bfe3678124b09cf6f9'], gamepad => [0, 'ad63dc64f4bdafb503f7015d04e8849cef5d99b4'], graphicaleffects => [0, '4119e4e6dc94447d773a01c1d6e4de0fefb9235a'], imageformats => [6, 'abe44c0f526e499668b1131c5990d9b571f99c46'], location => [3, 'f991e28cb0a670597f1955585c76ce8a26ce9e4b'], lottie => [0, '56f94cb8e2da9801ada7aa06f86ccf807f5a4ed8'], multimedia => [1, '0d7cc33ac1404758886acef4f804b788c6774e98'], networkauth => [0, 'f082a4c84c54e888b8d023ba68b7085551403425'], purchasing => [0, '9dfea35b04dcb52d02d7a2883df88d89ba9999ef'], quick3d => [1, '47defc8b33b7bdf1dbf289b65b301fba2def9b1c'], quickcontrols => [0, '18977875d16e22ad68a1dc2d7ee0a9c9f873c941'], quickcontrols2 => [7, '56ce8233382a091a8476c831edd416b5f704ae4f'], quicktimeline => [0, '4cd0142a30bfa5eef47c720ac24dd73e12764806'], remoteobjects => [0, '929c7ad0676f084b9ecc469cd47a307596923cb3'], scxml => [0, '0c93f94a44e2dce7eed9d17d4976b0c1e14be7bb'], sensors => [0, '6add85fa1a234a7e1943ba175c6fc799ccbae48e'], serialbus => [0, 'ced5c7223d037aece1e7f37d4314f388252de025'], serialport => [0, 'e2851096dc6f6a7cfa635d69ea950b382e3658ab'], speech => [1, '255845e2b2e605363762be25932d92fc10d32749'], svg => [8, 'a7a0f2491334e8cb4ef5731f5eb741f3f7b9af76'], tools => [1, '090e526e713d01eac34c64e4a09ad961c612febf'], translations => [0, 'af8cd030fed6a47cc1e8727e7ee5445e037bf712'], virtualkeyboard => [0, '5f66c9571303170f07954f73b09cad4cee1ce5d0'], wayland => [56, '2904e1b3c3004153b49c4dabfec04cc1ff5e3284'], webchannel => [3, 'f8949655ccfacc2d34cfb0af23c540db84a2b9e5'], webglplugin => [0, '655be6c5406f8ba42acaca363fc55d78a6198733'], websockets => [2, '63fb8da1ecf8e48262cd515690cf71a425f92bf5'], webview => [0, 'dfd86e07019488954cddcf2ba314df3cd0c01c0c'], x11extras => [0, 'e44c85e8643f2724109993a7b9eaf0dff3530fec'], xmlpatterns => [0, '3199d91de3f38e5ece3d36bcefe2c33b2c014f3f'], ); # $rev := git rev-list --count v5.15.2..$commit_on_kde_fork # find templates; populate "pages" array my @pages; my $template_file_name = 'PKGBUILD.sh.ep'; my $top_level_dirs = $pkgbuilds_dir->list({dir => 1}); for my $top_level_dir (@$top_level_dirs) { next unless -d $top_level_dir; next unless $top_level_dir ne 'devel'; my $default_package_name = $top_level_dir->basename; my ($qt_module, $qt_major_version); if ($default_package_name =~ qr/qt(5|6)-(.*)/) { $qt_major_version = $1; $qt_module = $2; } my $kde_fork_revision; if ($qt_major_version && $qt_major_version eq '5' && $qt_module) { $kde_fork_revision = $kde_fork_revisions{$qt_module}; } my $variant_dirs = $top_level_dir->list({dir => 1}); for my $variant_dir (@$variant_dirs) { next unless -d $variant_dir; my $variant = $variant_dir->basename; my $template_file = $variant_dir->child($template_file_name); if (!-f $template_file) { # print warning; all additional Qt repos for mingw-w64 should be converted to use templates now $log->warn("No template $template_file_name present for Qt module $qt_module and variant $variant") if defined $qt_module && index($variant, 'mingw-w64') == 0 && $variant !~ qr/.*-test$/; next; } # determine files my $files = $variant_dir->list; my $patch_files = $files->grep(qr/.*\.patch/); my $qt_module_sha256_file_name = "qt$qt_module-sha256.txt"; my $qt_module_sha256_file = defined $qt_module ? $variant_dir->child($qt_module_sha256_file_name) : undef; my $qt_module_sha256 = defined $qt_module_sha256_file && -f $qt_module_sha256_file ? Mojo::Util::trim($qt_module_sha256_file->slurp) : "$qt_module_sha256_file_name missing"; # determine variant parts my $variant_prefix_part = $variant; my $variant_suffix_part = ''; if ($variant) { for my $variant_suffixes (@variant_suffixes) { for my $variant_suffix (@$variant_suffixes) { next unless $variant =~ qr/.*-$variant_suffix$/; $variant_prefix_part = substr($variant, 0, length($variant) - length($variant_suffix) - 1); $variant_suffix_part = $variant_suffix_part ? "$variant_suffix-$variant_suffix_part" : $variant_suffix; last; } } } my $package_name_prefix = $variant_prefix_part ? "$variant_prefix_part-" : ""; my $package_name_suffix = $variant_suffix_part ? "-$variant_suffix_part" : ""; my $is_static_variant = $variant_suffix_part =~ qr/static/; my $has_static_variant = $is_static_variant || -d "$default_package_name/$variant-static"; my $is_mingw = $package_name_prefix eq 'mingw-w64-'; my $package_name = "$package_name_prefix$default_package_name$package_name_suffix"; next if defined $filter_regex && $package_name !~ $filter_regex; push(@pages, { install_path => "$default_package_name/$variant/PKGBUILD", template_params => [ template => "$default_package_name/$variant/PKGBUILD", stash_variables => { variant => $variant, variant_prefix_part => $variant_prefix_part, variant_suffix_part => $variant_suffix_part, default_package_name => $default_package_name, package_name_prefix => $package_name_prefix, package_name_suffix => $package_name_suffix, package_name => $package_name, files => $files, patch_files => $patch_files, qt_major_version => $qt_major_version, qt_module => $qt_module, qt_module_sha256 => $qt_module_sha256, kde_fork_revision => $kde_fork_revision, static_variant => $is_static_variant, static_suffix => $is_static_variant ? '-static' : '', static_deps => undef, static_makedeps => undef, is_mingw => $is_mingw, library_extension => $is_static_variant ? 'a' : ($is_mingw ? 'dll.a' : 'so'), install_prefix => $is_mingw ? '/usr/$_arch' : '/usr', shared_config => !$is_static_variant, static_config => $is_static_variant || !$has_static_variant, no_libraries => 0, }, ] }); } } # render "pages" for my $page (@pages) { # process template params my $template_params = $page->{template_params}; my $template_source_path; my $template_target_path; my $template_stash_variables; if (defined $template_params) { my ($template_name, $template_args) = (@$template_params % 2 ? shift @$template_params : undef, {@$template_params}); my $template_format = ($template_args->{format} //= 'sh'); my $template_handler = $template_args->{handler} // 'ep'; $template_name //= $template_args->{template}; $template_stash_variables = delete $template_args->{stash_variables}; $template_source_path = "$template_name.$template_format.$template_handler"; $template_target_path = "$template_name.$template_format"; $template_params = $template_args; $template_params->{template} = $template_name; } # determine source path and target path my $source_path = $page->{source_path} // $template_source_path; if (!$template_params && !$source_path) { die 'page needs either template_params or source_path'; } my $output_file = path($install_directory, $page->{install_path} // $template_target_path // $source_path); # ensure output directory exists $output_file->dirname->make_path unless -f $output_file; # skip unless the target is outdated # note: Can not skip templates that easy because that would require tracking includes. if (!defined $template_params && !is_outdated($source_path, $output_file)) { $log->info("Skipping '$source_path' -> '$output_file' (target up-to-date)"); next; } # do a simple copy if (!defined $template_params) { $log->info("Copying '$source_path' -> '$output_file'"); Mojo::File->new($source_path)->copy_to($output_file); next; } # render template $log->info("Rendering '$source_path' -> '$output_file'"); my $controller = $mojolicious->build_controller; $controller->stash($template_stash_variables) if defined $template_stash_variables; my $output = $controller->render_to_string(%$template_params); $log->debug($output); $output_file->spurt(encode('UTF-8', $output)); }