summaryrefslogtreecommitdiffstats
path: root/Bugzilla/Install
diff options
context:
space:
mode:
authorDylan William Hardison <dylan@hardison.net>2017-07-03 21:38:38 +0200
committerDylan William Hardison <dylan@hardison.net>2017-07-07 00:19:20 +0200
commit6f68125893590fc9de60185f5535bae12adbcb54 (patch)
tree6d7b2b7885a750826655cf3896d7fd72791623b3 /Bugzilla/Install
parente3e2c7c0273499f832ee692ca63620cd8aa8bda1 (diff)
downloadbugzilla-6f68125893590fc9de60185f5535bae12adbcb54.tar.gz
bugzilla-6f68125893590fc9de60185f5535bae12adbcb54.tar.xz
Bug 1377232 - Revert code from bug 1361890
Diffstat (limited to 'Bugzilla/Install')
-rw-r--r--Bugzilla/Install/AssetManager.pm218
-rw-r--r--Bugzilla/Install/Filesystem.pm42
2 files changed, 27 insertions, 233 deletions
diff --git a/Bugzilla/Install/AssetManager.pm b/Bugzilla/Install/AssetManager.pm
deleted file mode 100644
index 073e012cd..000000000
--- a/Bugzilla/Install/AssetManager.pm
+++ /dev/null
@@ -1,218 +0,0 @@
-# This Source Code Form is subject to the terms of the Mozilla Public
-# License, v. 2.0. If a copy of the MPL was not distributed with this
-# file, You can obtain one at http://mozilla.org/MPL/2.0/.
-#
-# This Source Code Form is "Incompatible With Secondary Licenses", as
-# defined by the Mozilla Public License, v. 2.0.
-
-package Bugzilla::Install::AssetManager;
-
-use 5.10.1;
-use strict;
-use warnings;
-
-use Moo;
-use MooX::StrictConstructor;
-use Type::Utils;
-use Types::Standard qw(Bool Str ArrayRef);
-
-use Digest::SHA ();
-use File::Copy qw(cp);
-use File::Find qw(find);
-use File::Basename qw(dirname);
-use File::Spec;
-use JSON::XS ();
-use MIME::Base64 qw(encode_base64);
-use File::Slurp;
-use List::MoreUtils qw(any all);
-use Carp;
-
-use Bugzilla::Constants qw(bz_locations);
-
-our $VERSION = 1;
-
-my $SHA_VERSION = '224';
-
-my $ABSOLUTE_DIR = declare as Str, where { File::Spec->file_name_is_absolute($_) && -d $_ }
-message {"must be an absolute path to a directory"};
-
-has 'base_dir' => ( is => 'lazy', isa => $ABSOLUTE_DIR );
-has 'asset_dir' => ( is => 'lazy', isa => $ABSOLUTE_DIR );
-has 'source_dirs' => ( is => 'lazy' );
-has 'state' => ( is => 'lazy' );
-has 'state_file' => ( is => 'lazy' );
-has 'json' => ( is => 'lazy' );
-
-sub asset_file {
- my ( $self, $file, $relative_to ) = @_;
- $relative_to //= $self->base_dir;
- my $asset_file = $self->state->{asset_map}->{$file}
- or return $file;
-
- return File::Spec->abs2rel( File::Spec->catfile( $self->asset_dir, $asset_file ), $relative_to );
-}
-
-sub asset_files {
- my ( $self, @files ) = @_;
-
- return \@files;
-}
-
-sub asset_sri {
- my ( $self, $asset_file ) = @_;
- my ($hex) = $asset_file =~ m!([[:xdigit:]]+)\.\w+$!;
- my $data = pack "H*", $hex;
- return "sha$SHA_VERSION-" . encode_base64( $data, "" );
-}
-
-sub compile_file {
- my ( $self, $file ) = @_;
- return unless -f $file;
- my $base_dir = $self->base_dir;
- my $asset_dir = $self->asset_dir;
- my $asset_map = $self->state->{asset_map};
-
- my $key = File::Spec->abs2rel( $file, $base_dir );
- return if $asset_map->{$key};
-
- if ( $file =~ /\.(jpe?g|png|gif|ico|woff|js)$/i ) {
- my $ext = $1;
- my $digest = $self->_digest_file($file) or die "invalid digest for $file";
- my $asset_file = File::Spec->catfile( $asset_dir, "$digest.$ext" );
- cp( $file, $asset_file ) or die "failed to copy $file to $asset_file: $!";
- if ( $digest eq $self->_digest_file($asset_file) ) {
- $asset_map->{$key} = File::Spec->abs2rel( $asset_file, $asset_dir );
- }
- else {
- die "failed to write $asset_file";
- }
- }
- elsif ( $file =~ /\.css$/ ) {
- my $content = read_file($file);
-
- $content =~ s{(?<!=)url\(([^\)]+)\)}{$self->_css_url_rewrite($1, $file)}eig;
- my $digest = $self->_digest_string($content);
- my $asset_file = File::Spec->catfile( $asset_dir, "$digest.css" );
- write_file( $asset_file, $content ) or die "failed to write $asset_file: $!";
- if ( $digest eq $self->_digest_file($asset_file) ) {
- $asset_map->{$key} = File::Spec->abs2rel( $asset_file, $asset_dir );
- }
- else {
- die "failed to write $asset_file";
- }
- }
-}
-
-sub compile_all {
- my ($self) = @_;
- my $asset_map = $self->state->{asset_map} = {};
-
- my $wanted = sub {
- $self->compile_file($File::Find::name);
- };
-
- find( { wanted => $wanted, no_chdir => 1 }, @{ $self->source_dirs } );
-
- $self->_save_state();
-}
-
-sub _css_url_rewrite {
- my ( $self, $url, $file ) = @_;
- my $dir = dirname($file);
-
- # rewrite relative urls as the unified stylesheet lives in a different
- # directory from the source
- $url =~ s/(^['"]|['"]$)//g;
- if ( $url =~ m!^(/|data:)! ) {
- return 'url(' . $url . ')';
- }
- else {
- my $url_file = File::Spec->rel2abs( $url, $dir );
- my $ref_file = File::Spec->abs2rel( $url_file, $self->base_dir );
- $self->compile_file($url_file);
- return sprintf( "url(%s)", $self->asset_file( $ref_file, $self->asset_dir ) );
- }
-}
-
-sub _new_digest { Digest::SHA->new($SHA_VERSION) }
-
-sub _digest_file {
- my ( $self, $file ) = @_;
- my $digest = $self->_new_digest;
- $digest->addfile( $file, "b" );
- return $digest->hexdigest;
-}
-
-sub _digest_string {
- my ( $self, $string ) = @_;
- my $digest = $self->_new_digest;
- $digest->add($string);
- return $digest->hexdigest;
-}
-
-sub _build_base_dir {
- Cwd::realpath( File::Spec->rel2abs( bz_locations->{cgi_path} ) );
-}
-
-sub _build_asset_dir {
- my ($self) = @_;
- my $dir
- = Cwd::realpath( File::Spec->rel2abs( bz_locations->{assetsdir} ) );
-
- if ( $dir && -d $dir ) {
- my $version_dir = File::Spec->catdir( $dir, "v" . $self->VERSION );
- unless ( -d $version_dir ) {
- mkdir $version_dir or die "mkdir $version_dir failed: $!";
- }
- return $version_dir;
- }
- else {
- return $dir;
- }
-}
-
-sub _build_source_dirs {
- my ($self) = @_;
- my $base = $self->base_dir;
- my $ext_dir = "$base/extensions";
- opendir my $ext_dir_handle, $ext_dir or die "unable to open $ext_dir: $!";
- my @dirs = grep { -d $_ } map {"$ext_dir/$_/web"} grep { !/^\.\.?$/ } readdir $ext_dir_handle;
- closedir $ext_dir_handle;
-
- return [ "$base/images", "$base/skins", "$base/js", grep { -d $_ } @dirs ];
-}
-
-sub _build_state_file {
- my ($self) = @_;
- return $self->asset_dir . "/state.json";
-}
-
-sub _build_state {
- my ($self) = @_;
- my $state;
- if ( open my $fh, '<:bytes', $self->state_file ) {
- local $/ = undef;
- my $json = <$fh>;
- close $fh;
- $state = $self->json->decode($json);
- }
- else {
- $state = {};
- }
-
- $state->{asset_map} //= {};
-
- return $state;
-}
-
-sub _build_json { JSON::XS->new->canonical->utf8 }
-
-sub _save_state {
- my ($self) = @_;
- open my $fh, '>:bytes', $self->state_file
- or die "unable to write state file: $!";
- print $fh $self->json->encode( $self->state );
- close $fh;
-}
-
-1;
diff --git a/Bugzilla/Install/Filesystem.pm b/Bugzilla/Install/Filesystem.pm
index 162e324f7..22ec34a95 100644
--- a/Bugzilla/Install/Filesystem.pm
+++ b/Bugzilla/Install/Filesystem.pm
@@ -31,7 +31,6 @@ use File::Path;
use File::Basename;
use File::Copy qw(move);
use File::Spec;
-use File::stat;
use Cwd ();
use File::Slurp;
use IO::File;
@@ -82,16 +81,12 @@ EOT
use constant HT_ASSETS_DIR => <<'EOT';
# Allow access to .css and js files
-<FilesMatch state\.json$>
- Deny from all
+<FilesMatch \.(css|js)$>
+ Allow from all
</FilesMatch>
-FileETag None
-Header set Cache-Control "public, immutable, max-age=31536000"
-Header set Content-Security-Policy "default-src 'none';"
-
# And no directory listings, either.
-Options -Indexes
+Deny from all
EOT
use constant INDEX_HTML => <<'EOT';
@@ -349,7 +344,7 @@ sub FILESYSTEM {
$attachdir => DIR_CGI_WRITE,
$graphsdir => DIR_CGI_WRITE | DIR_ALSO_WS_SERVE,
$webdotdir => DIR_CGI_WRITE | DIR_ALSO_WS_SERVE,
- $assetsdir => DIR_WS_SERVE,
+ $assetsdir => DIR_CGI_WRITE | DIR_ALSO_WS_SERVE,
$template_cache => DIR_CGI_WRITE,
$error_reports => DIR_CGI_WRITE,
# Directories that contain content served directly by the web server.
@@ -451,13 +446,8 @@ sub FILESYSTEM {
"$webdotdir/.htaccess" => { perms => WS_SERVE,
contents => HT_WEBDOT_DIR },
"$assetsdir/.htaccess" => { perms => WS_SERVE,
- contents => HT_ASSETS_DIR },
+ contents => HT_ASSETS_DIR },
);
- my $mtime = stat(__FILE__)->mtime;
- foreach my $file (keys %htaccess) {
- my $file_stat = stat($file);
- $htaccess{$file}{overwrite} = $file_stat && $mtime > $file_stat->mtime;
- }
Bugzilla::Hook::process('install_filesystem', {
files => \%files,
@@ -571,6 +561,7 @@ sub update_filesystem {
_remove_empty_css_files();
_convert_single_file_skins();
+ _remove_dynamic_assets();
}
sub _css_url_fix {
@@ -636,6 +627,27 @@ sub _convert_single_file_skins {
}
}
+# delete all automatically generated css/js files to force recreation at the
+# next request.
+sub _remove_dynamic_assets {
+ my @files = (
+ glob(bz_locations()->{assetsdir} . '/*.css'),
+ glob(bz_locations()->{assetsdir} . '/*.js'),
+ );
+ foreach my $file (@files) {
+ unlink($file);
+ }
+
+ # remove old skins/assets directory
+ my $old_path = bz_locations()->{skinsdir} . '/assets';
+ if (-d $old_path) {
+ foreach my $file (glob("$old_path/*.css")) {
+ unlink($file);
+ }
+ rmdir($old_path);
+ }
+}
+
sub create_htaccess {
_create_files(%{FILESYSTEM()->{htaccess}});