2008-11-30 09:26:05 +00:00
|
|
|
<?php
|
|
|
|
/* -*- tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
|
|
/*
|
|
|
|
# ***** BEGIN LICENSE BLOCK *****
|
|
|
|
# This file is part of InDefero, an open source project management application.
|
|
|
|
# Copyright (C) 2008 Céondo Ltd and contributors.
|
|
|
|
#
|
|
|
|
# InDefero is free software; you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation; either version 2 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# InDefero is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program; if not, write to the Free Software
|
|
|
|
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
#
|
|
|
|
# ***** END LICENSE BLOCK ***** */
|
|
|
|
|
|
|
|
Pluf::loadFunction('Pluf_HTTP_URL_urlForView');
|
|
|
|
Pluf::loadFunction('Pluf_Shortcuts_RenderToResponse');
|
|
|
|
Pluf::loadFunction('Pluf_Shortcuts_GetObjectOr404');
|
|
|
|
Pluf::loadFunction('Pluf_Shortcuts_GetFormForModel');
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Review views.
|
|
|
|
*/
|
|
|
|
class IDF_Views_Review
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* View list of reviews for a given project.
|
|
|
|
*/
|
|
|
|
public $index_precond = array('IDF_Precondition::accessReview');
|
|
|
|
public function index($request, $match)
|
|
|
|
{
|
|
|
|
$prj = $request->project;
|
|
|
|
$title = sprintf(__('%s Code Reviews'), (string) $prj);
|
|
|
|
// Paginator to paginate the pages
|
|
|
|
$pag = new Pluf_Paginator(new IDF_Review());
|
|
|
|
$pag->class = 'recent-issues';
|
|
|
|
$pag->item_extra_props = array('project_m' => $prj,
|
|
|
|
'shortname' => $prj->shortname,
|
|
|
|
'current_user' => $request->user);
|
|
|
|
$pag->summary = __('This table shows the latest reviews.');
|
|
|
|
$pag->action = array('IDF_Views_Review::index', array($prj->shortname));
|
|
|
|
$otags = $prj->getTagIdsByStatus('open');
|
|
|
|
if (count($otags) == 0) $otags[] = 0;
|
|
|
|
$pag->forced_where = new Pluf_SQL('project=%s AND status IN ('.implode(', ', $otags).')', array($prj->id));
|
|
|
|
$pag->action = array('IDF_Views_Issue::index', array($prj->shortname));
|
|
|
|
$pag->sort_order = array('modif_dtime', 'ASC'); // will be reverted
|
|
|
|
$pag->sort_reverse_order = array('modif_dtime');
|
|
|
|
$list_display = array(
|
|
|
|
'id' => __('Id'),
|
|
|
|
array('summary', 'IDF_Views_Review_SummaryAndLabels', __('Summary')),
|
|
|
|
array('status', 'IDF_Views_Issue_ShowStatus', __('Status')),
|
|
|
|
array('modif_dtime', 'Pluf_Paginator_DateAgo', __('Last Updated')),
|
|
|
|
);
|
|
|
|
$pag->configure($list_display, array(), array('title', 'modif_dtime'));
|
|
|
|
$pag->items_per_page = 25;
|
|
|
|
$pag->no_results_text = __('No reviews were found.');
|
|
|
|
$pag->sort_order = array('modif_dtime', 'ASC');
|
|
|
|
$pag->setFromRequest($request);
|
|
|
|
return Pluf_Shortcuts_RenderToResponse('idf/review/index.html',
|
|
|
|
array(
|
|
|
|
'page_title' => $title,
|
|
|
|
'reviews' => $pag,
|
|
|
|
),
|
|
|
|
$request);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new code review.
|
|
|
|
*/
|
|
|
|
public $create_precond = array('IDF_Precondition::accessReview',
|
|
|
|
'Pluf_Precondition::loginRequired');
|
|
|
|
public function create($request, $match)
|
|
|
|
{
|
|
|
|
$prj = $request->project;
|
|
|
|
$title = __('Start Code Review');
|
|
|
|
if ($request->method == 'POST') {
|
|
|
|
$form = new IDF_Form_ReviewCreate(array_merge($request->POST,
|
|
|
|
$request->FILES),
|
|
|
|
array('project' => $prj,
|
|
|
|
'user' => $request->user
|
|
|
|
));
|
|
|
|
if ($form->isValid()) {
|
|
|
|
$review = $form->save();
|
2010-12-05 00:22:32 +00:00
|
|
|
$urlr = Pluf_HTTP_URL_urlForView('IDF_Views_Review::view',
|
2008-11-30 09:26:05 +00:00
|
|
|
array($prj->shortname, $review->id));
|
|
|
|
$request->user->setMessage(sprintf(__('The <a href="%s">code review %d</a> has been created.'), $urlr, $review->id));
|
2010-12-05 00:22:32 +00:00
|
|
|
$url = Pluf_HTTP_URL_urlForView('IDF_Views_Review::index',
|
2008-11-30 09:26:05 +00:00
|
|
|
array($prj->shortname));
|
|
|
|
return new Pluf_HTTP_Response_Redirect($url);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$form = new IDF_Form_ReviewCreate(null,
|
|
|
|
array('project' => $prj,
|
|
|
|
'user' => $request->user));
|
|
|
|
}
|
|
|
|
return Pluf_Shortcuts_RenderToResponse('idf/review/create.html',
|
|
|
|
array(
|
|
|
|
'page_title' => $title,
|
|
|
|
'form' => $form,
|
|
|
|
),
|
|
|
|
$request);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Download the patch of a review.
|
|
|
|
*/
|
|
|
|
public $getPatch_precond = array('IDF_Precondition::accessReview');
|
|
|
|
public function getPatch($request, $match)
|
|
|
|
{
|
|
|
|
$prj = $request->project;
|
|
|
|
$patch = Pluf_Shortcuts_GetObjectOr404('IDF_Review_Patch', $match[2]);
|
|
|
|
$prj->inOr404($patch->get_review());
|
|
|
|
$file = Pluf::f('upload_issue_path').'/'.$patch->patch;
|
|
|
|
|
|
|
|
$rep = new Pluf_HTTP_Response_File($file, 'text/plain');
|
|
|
|
$rep->headers['Content-Disposition'] = 'attachment; filename="'.$patch->id.'.diff"';
|
|
|
|
return $rep;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* View a code review.
|
|
|
|
*/
|
|
|
|
public $view_precond = array('IDF_Precondition::accessReview');
|
|
|
|
public function view($request, $match)
|
|
|
|
{
|
|
|
|
$prj = $request->project;
|
|
|
|
$review = Pluf_Shortcuts_GetObjectOr404('IDF_Review', $match[2]);
|
|
|
|
$prj->inOr404($review);
|
|
|
|
$url = Pluf_HTTP_URL_urlForView('IDF_Views_Review::view',
|
|
|
|
array($prj->shortname, $review->id));
|
|
|
|
$title = Pluf_Template::markSafe(sprintf(__('Review <a href="%s">%d</a>: %s'), $url, $review->id, $review->summary));
|
|
|
|
|
|
|
|
$patches = $review->get_patches_list();
|
|
|
|
$patch = $patches[0];
|
|
|
|
$diff = new IDF_Diff(file_get_contents(Pluf::f('upload_issue_path').'/'.$patch->patch));
|
|
|
|
$diff->parse();
|
|
|
|
// The form to submit comments is based on the files in the
|
|
|
|
// diff
|
|
|
|
if ($request->method == 'POST' and !$request->user->isAnonymous()) {
|
|
|
|
$form = new IDF_Form_ReviewFileComment($request->POST,
|
|
|
|
array('files' => $diff->files,
|
|
|
|
'user' => $request->user,
|
|
|
|
'patch' => $patch,
|
2009-10-29 12:25:50 +00:00
|
|
|
'project' => $prj,
|
2008-11-30 09:26:05 +00:00
|
|
|
));
|
|
|
|
if ($form->isValid()) {
|
2009-10-08 09:23:33 +00:00
|
|
|
$review_comment = $form->save();
|
2008-12-01 20:39:17 +00:00
|
|
|
$review = $patch->get_review();
|
2010-12-05 00:22:32 +00:00
|
|
|
$urlr = Pluf_HTTP_URL_urlForView('IDF_Views_Review::view',
|
2008-12-01 20:39:17 +00:00
|
|
|
array($prj->shortname, $review->id));
|
|
|
|
$request->user->setMessage(sprintf(__('Your <a href="%s">code review %d</a> has been published.'), $urlr, $review->id));
|
2010-12-05 00:22:32 +00:00
|
|
|
$url = Pluf_HTTP_URL_urlForView('IDF_Views_Review::index',
|
2008-11-30 09:26:05 +00:00
|
|
|
array($prj->shortname));
|
2009-10-08 09:23:33 +00:00
|
|
|
$review_comment->notify($request->conf);
|
2008-11-30 09:26:05 +00:00
|
|
|
return new Pluf_HTTP_Response_Redirect($url);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$form = new IDF_Form_ReviewFileComment(null,
|
|
|
|
array('files' => $diff->files,
|
|
|
|
'user' => $request->user,
|
2009-10-29 12:25:50 +00:00
|
|
|
'project' => $prj,
|
2008-11-30 09:26:05 +00:00
|
|
|
'patch' => $patch,));
|
|
|
|
}
|
2008-12-05 10:34:02 +00:00
|
|
|
$scm = IDF_Scm::get($request->project);
|
2008-11-30 09:26:05 +00:00
|
|
|
$files = array();
|
|
|
|
$reviewers = array();
|
|
|
|
foreach ($diff->files as $filename => $def) {
|
2009-05-30 18:04:43 +00:00
|
|
|
$fileinfo = $scm->getPathInfo($filename, $patch->get_commit()->scm_id);
|
2008-11-30 09:26:05 +00:00
|
|
|
$sql = new Pluf_SQL('cfile=%s', array($filename));
|
2009-07-16 08:04:58 +00:00
|
|
|
$cts = $patch->getFileComments(array('filter'=>$sql->gen(),
|
|
|
|
'order'=>'creation_dtime ASC'));
|
2008-11-30 09:26:05 +00:00
|
|
|
foreach ($cts as $ct) {
|
2009-07-16 08:04:58 +00:00
|
|
|
$reviewers[] = $ct->get_comment()->get_submitter();
|
2008-11-30 09:26:05 +00:00
|
|
|
}
|
2010-12-05 00:22:32 +00:00
|
|
|
if (count($def['chunks'])) {
|
2009-05-30 18:04:43 +00:00
|
|
|
$orig_file = ($fileinfo) ? $scm->getFile($fileinfo) : '';
|
2008-11-30 09:26:05 +00:00
|
|
|
$files[$filename] = array(
|
|
|
|
$diff->fileCompare($orig_file, $def, $filename),
|
|
|
|
$form->f->{md5($filename)},
|
|
|
|
$cts,
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
$files[$filename] = array('', $form->f->{md5($filename)}, $cts);
|
|
|
|
}
|
|
|
|
}
|
2010-12-05 00:22:32 +00:00
|
|
|
|
2008-11-30 09:26:05 +00:00
|
|
|
$reviewers = Pluf_Model_RemoveDuplicates($reviewers);
|
|
|
|
return Pluf_Shortcuts_RenderToResponse('idf/review/view.html',
|
2009-10-29 12:25:50 +00:00
|
|
|
array_merge(
|
2008-11-30 09:26:05 +00:00
|
|
|
array(
|
|
|
|
'page_title' => $title,
|
|
|
|
'review' => $review,
|
|
|
|
'files' => $files,
|
|
|
|
'diff' => $diff,
|
|
|
|
'patch' => $patch,
|
2009-08-03 19:23:23 +00:00
|
|
|
'comments' => $patch->get_comments_list(array('sort' => 'id ASC')),
|
2008-11-30 09:26:05 +00:00
|
|
|
'form' => $form,
|
|
|
|
'reviewers' => $reviewers,
|
|
|
|
),
|
2009-10-29 12:25:50 +00:00
|
|
|
IDF_Views_Issue::autoCompleteArrays($prj)
|
|
|
|
),
|
2008-11-30 09:26:05 +00:00
|
|
|
$request);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Display the summary of an review, then on a new line, display the
|
|
|
|
* list of labels with a link to a view "by label only".
|
|
|
|
*
|
|
|
|
* The summary of the review is linking to the review.
|
|
|
|
*/
|
|
|
|
function IDF_Views_Review_SummaryAndLabels($field, $review, $extra='')
|
|
|
|
{
|
2010-12-05 00:22:32 +00:00
|
|
|
$edit = Pluf_HTTP_URL_urlForView('IDF_Views_Review::view',
|
2008-11-30 09:26:05 +00:00
|
|
|
array($review->shortname, $review->id));
|
|
|
|
$tags = array();
|
|
|
|
foreach ($review->get_tags_list() as $tag) {
|
|
|
|
$tags[] = Pluf_esc($tag);
|
|
|
|
}
|
|
|
|
$out = '';
|
|
|
|
if (count($tags)) {
|
|
|
|
$out = '<br /><span class="label note">'.implode(', ', $tags).'</span>';
|
|
|
|
}
|
|
|
|
return sprintf('<a href="%s">%s</a>', $edit, Pluf_esc($review->summary)).$out;
|
|
|
|
}
|
|
|
|
|