aboutsummaryrefslogtreecommitdiffstats
path: root/phpBB/phpbb/db/migration/data/v310/softdelete_p2.php
blob: 849a996c1b3d715e49b4f1ec245af55d1c1249e4 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
<?php
/**
*
* This file is part of the phpBB Forum Software package.
*
* @copyright (c) phpBB Limited <https://www.phpbb.com>
* @license GNU General Public License, version 2 (GPL-2.0)
*
* For full copyright and license information, please see
* the docs/CREDITS.txt file.
*
*/

namespace phpbb\db\migration\data\v310;

class softdelete_p2 extends \phpbb\db\migration\migration
{
	public function effectively_installed()
	{
		return !$this->db_tools->sql_column_exists($this->table_prefix . 'posts', 'post_approved');
	}

	static public function depends_on()
	{
		return array(
			'\phpbb\db\migration\data\v310\dev',
			'\phpbb\db\migration\data\v310\softdelete_p1',
		);
	}

	public function update_schema()
	{
		return array(
			'drop_columns'		=> array(
				$this->table_prefix . 'forums'			=> array('forum_posts', 'forum_topics', 'forum_topics_real'),
				$this->table_prefix . 'posts'			=> array('post_approved'),
				$this->table_prefix . 'topics'			=> array('topic_approved', 'topic_replies', 'topic_replies_real'),
			),
			'drop_keys'		=> array(
				$this->table_prefix . 'posts'			=> array('post_approved'),
				$this->table_prefix . 'topics'			=> array(
					'forum_appr_last',
					'topic_approved',
				),
			),
		);
	}

	public function revert_schema()
	{
		return array(
			'add_columns'		=> array(
				$this->table_prefix . 'forums'			=> array(
					'forum_posts'			=> array('UINT', 0),
					'forum_topics'			=> array('UINT', 0),
					'forum_topics_real'		=> array('UINT', 0),
				),
				$this->table_prefix . 'posts'			=> array(
					'post_approved'			=> array('BOOL', 1),
				),
				$this->table_prefix . 'topics'		=> array(
					'topic_approved'			=> array('BOOL', 1),
					'topic_replies'				=> array('UINT', 0),
					'topic_replies_real'		=> array('UINT', 0),
				),
			),
			'add_index'		=> array(
				$this->table_prefix . 'posts'			=> array(
					'post_approved'		=> array('post_approved'),
				),
				$this->table_prefix . 'topics'		=> array(
					'forum_appr_last'	=> array('forum_id', 'topic_approved', 'topic_last_post_id'),
					'topic_approved'	=> array('topic_approved'),
				),
			),
		);
	}
}
pan> # # Date handling # sub date_adjust_down { my ($year, $month, $day) = @_; if ($day == 0) { $month -= 1; $day = 31; # Proper day adjustment is done later. if ($month == 0) { $year -= 1; $month = 12; } } if (($month == 2) && ($day > 28)) { if ($year % 4 == 0 && $year % 100 != 0) { $day = 29; } else { $day = 28; } } if (($month == 4 || $month == 6 || $month == 9 || $month == 11) && ($day == 31) ) { $day = 30; } return ($year, $month, $day); } sub date_adjust_up { my ($year, $month, $day) = @_; if ($day > 31) { $month += 1; $day = 1; if ($month == 13) { $month = 1; $year += 1; } } if ($month == 2 && $day > 28) { if ($year % 4 != 0 || $year % 100 == 0 || $day > 29) { $month = 3; $day = 1; } } if (($month == 4 || $month == 6 || $month == 9 || $month == 11) && ($day == 31) ) { $month += 1; $day = 1; } return ($year, $month, $day); } sub split_by_month { # Takes start and end dates and splits them into a list of # monthly-spaced 2-lists of dates. my ($start_date, $end_date) = @_; # We assume at this point that the dates are provided and sane my (undef, undef, undef, $sd, $sm, $sy, undef) = strptime($start_date); my (undef, undef, undef, $ed, $em, $ey, undef) = strptime($end_date); # Find out how many months fit between the two dates so we know # how many times we loop. my $yd = $ey - $sy; my $md = 12 * $yd + $em - $sm; # If the end day is smaller than the start day, last interval is not a whole month. if ($sd > $ed) { $md -= 1; } my (@months, $sub_start, $sub_end); # This +1 and +1900 are a result of strptime's bizarre semantics my $year = $sy + 1900; my $month = $sm + 1; # Keep the original date, when the date will be changed in the adjust_date. my $sd_tmp = $sd; my $month_tmp = $month; my $year_tmp = $year; # This section handles only the whole months. for (my $i=0; $i < $md; $i++) { # Start of interval is adjusted up: 31.2. -> 1.3. ($year_tmp, $month_tmp, $sd_tmp) = date_adjust_up($year, $month, $sd); $sub_start = sprintf("%04d-%02d-%02d", $year_tmp, $month_tmp, $sd_tmp); $month += 1; if ($month == 13) { $month = 1; $year += 1; } # End of interval is adjusted down: 31.2 -> 28.2. ($year_tmp, $month_tmp, $sd_tmp) = date_adjust_down($year, $month, $sd - 1); $sub_end = sprintf("%04d-%02d-%02d", $year_tmp, $month_tmp, $sd_tmp); push @months, [$sub_start, $sub_end]; } # This section handles the last (unfinished) month. $sub_end = sprintf("%04d-%02d-%02d", $ey + 1900, $em + 1, $ed); ($year_tmp, $month_tmp, $sd_tmp) = date_adjust_up($year, $month, $sd); $sub_start = sprintf("%04d-%02d-%02d", $year_tmp, $month_tmp, $sd_tmp); push @months, [$sub_start, $sub_end]; return @months; } sub sqlize_dates { my ($start_date, $end_date) = @_; my $date_bits = ""; my @date_values; if ($start_date) { # we've checked, trick_taint is fine trick_taint($start_date); $date_bits = " AND longdescs.bug_when > ?"; push @date_values, $start_date; } if ($end_date) { # we need to add one day to end_date to catch stuff done today # do not forget to adjust date if it was the last day of month my (undef, undef, undef, $ed, $em, $ey, undef) = strptime($end_date); ($ey, $em, $ed) = date_adjust_up($ey+1900, $em+1, $ed+1); $end_date = sprintf("%04d-%02d-%02d", $ey, $em, $ed); $date_bits .= " AND longdescs.bug_when < ?"; push @date_values, $end_date; } return ($date_bits, \@date_values); } # Return all blockers of the current bug, recursively. sub get_blocker_ids { my ($bug_id, $unique) = @_; $unique ||= {$bug_id => 1}; my $deps = Bugzilla::Bug::EmitDependList("blocked", "dependson", $bug_id); my @unseen = grep { !$unique->{$_}++ } @$deps; foreach $bug_id (@unseen) { get_blocker_ids($bug_id, $unique); } return keys %$unique; } # Return a hashref whose key is chosen by the user (bug ID or commenter) # and value is a hash of the form {bug ID, commenter, time spent}. # So you can either view it as the time spent by commenters on each bug # or the time spent in bugs by each commenter. sub get_list { my ($bugids, $start_date, $end_date, $keyname) = @_; my $dbh = Bugzilla->dbh; my ($date_bits, $date_values) = sqlize_dates($start_date, $end_date); my $buglist = join(", ", @$bugids); # Returns the total time worked on each bug *per developer*. my $data = $dbh->selectall_arrayref( qq{SELECT SUM(work_time) AS total_time, login_name, longdescs.bug_id FROM longdescs INNER JOIN profiles ON longdescs.who = profiles.userid INNER JOIN bugs ON bugs.bug_id = longdescs.bug_id WHERE longdescs.bug_id IN ($buglist) $date_bits } . $dbh->sql_group_by('longdescs.bug_id, login_name', 'longdescs.bug_when') . qq{ HAVING SUM(work_time) > 0}, {Slice => {}}, @$date_values); my %list; # What this loop does is to push data having the same key in an array. push(@{$list{ $_->{$keyname} }}, $_) foreach @$data; return \%list; } # Return bugs which had no activity (a.k.a work_time = 0) during the given time range. sub get_inactive_bugs { my ($bugids, $start_date, $end_date) = @_; my $dbh = Bugzilla->dbh; my ($date_bits, $date_values) = sqlize_dates($start_date, $end_date); my $buglist = join(", ", @$bugids); my $bugs = $dbh->selectcol_arrayref( "SELECT bug_id FROM bugs WHERE bugs.bug_id IN ($buglist) AND NOT EXISTS ( SELECT 1 FROM longdescs WHERE bugs.bug_id = longdescs.bug_id AND work_time > 0 $date_bits)", undef, @$date_values); return $bugs; } # Return 1st day of the month of the earliest activity date for a given list of bugs. sub get_earliest_activity_date { my ($bugids) = @_; my $dbh = Bugzilla->dbh; my ($date) = $dbh->selectrow_array( 'SELECT ' . $dbh->sql_date_format('MIN(bug_when)', '%Y-%m-01') . ' FROM longdescs WHERE ' . $dbh->sql_in('bug_id', $bugids) . ' AND work_time > 0'); return $date; } # # Template code starts here # my $user = Bugzilla->login(LOGIN_REQUIRED); my $cgi = Bugzilla->cgi; my $template = Bugzilla->template; my $vars = {}; Bugzilla->switch_to_shadow_db(); $user->is_timetracker || ThrowUserError("auth_failure", {group => "time-tracking", action => "access", object => "timetracking_summaries"}); my @ids = split(",", $cgi->param('id') || ''); @ids = map { Bugzilla::Bug->check($_)->id } @ids; scalar(@ids) || ThrowUserError('no_bugs_chosen', {action => 'view'}); my $group_by = $cgi->param('group_by') || "number"; my $monthly = $cgi->param('monthly'); my $detailed = $cgi->param('detailed'); my $do_report = $cgi->param('do_report'); my $inactive = $cgi->param('inactive'); my $do_depends = $cgi->param('do_depends'); my $ctype = scalar($cgi->param("ctype")); my ($start_date, $end_date); if ($do_report) { my @bugs = @ids; # Dependency mode requires a single bug and grabs dependents. if ($do_depends) { if (scalar(@bugs) != 1) { ThrowCodeError("bad_arg", { argument=>"id", function=>"summarize_time"}); } @bugs = get_blocker_ids($bugs[0]); @bugs = grep { $user->can_see_bug($_) } @bugs; } $start_date = trim $cgi->param('start_date'); $end_date = trim $cgi->param('end_date'); foreach my $date ($start_date, $end_date) { next unless $date; validate_date($date) || ThrowUserError('illegal_date', {date => $date, format => 'YYYY-MM-DD'}); } # Swap dates in case the user put an end_date before the start_date if ($start_date && $end_date && str2time($start_date) > str2time($end_date)) { $vars->{'warn_swap_dates'} = 1; ($start_date, $end_date) = ($end_date, $start_date); } # Store dates in a session cookie so re-visiting the page # for other bugs keeps them around. $cgi->send_cookie(-name => 'time-summary-dates', -value => join ";", ($start_date, $end_date)); my (@parts, $part_data, @part_list); # Break dates apart into months if necessary; if not, we use the # same @parts list to allow us to use a common codepath. if ($monthly) { # Calculate the earliest activity date if the user doesn't # specify a start date. if (!$start_date) { $start_date = get_earliest_activity_date(\@bugs); } # Provide a default end date. Note that this differs in semantics # from the open-ended queries we use when start/end_date aren't # provided -- and clock skews will make this evident! @parts = split_by_month($start_date, $end_date || format_time(scalar localtime(time()), '%Y-%m-%d')); } else { @parts = ([$start_date, $end_date]); } # For each of the separate divisions, grab the relevant data. my $keyname = ($group_by eq 'owner') ? 'login_name' : 'bug_id'; foreach my $part (@parts) { my ($sub_start, $sub_end) = @$part; $part_data = get_list(\@bugs, $sub_start, $sub_end, $keyname); push(@part_list, $part_data); } # Do we want to see inactive bugs? if ($inactive) { $vars->{'null'} = get_inactive_bugs(\@bugs, $start_date, $end_date); } else { $vars->{'null'} = {}; } # Convert bug IDs to bug objects. @bugs = map {new Bugzilla::Bug($_)} @bugs; $vars->{'part_list'} = \@part_list; $vars->{'parts'} = \@parts; # We pass the list of bugs as a hashref. $vars->{'bugs'} = {map { $_->id => $_ } @bugs}; } elsif ($cgi->cookie("time-summary-dates")) { ($start_date, $end_date) = split ";", $cgi->cookie('time-summary-dates'); } $vars->{'ids'} = \@ids; $vars->{'start_date'} = $start_date; $vars->{'end_date'} = $end_date; $vars->{'group_by'} = $group_by; $vars->{'monthly'} = $monthly; $vars->{'detailed'} = $detailed; $vars->{'inactive'} = $inactive; $vars->{'do_report'} = $do_report; $vars->{'do_depends'} = $do_depends; my $format = $template->get_format("bug/summarize-time", undef, $ctype); # Get the proper content-type print $cgi->header(-type=> $format->{'ctype'}); $template->process("$format->{'template'}", $vars) || ThrowTemplateError($template->error());