Fix some issues caught by HipHop, and work around some issues

caused by HipHop.
This commit is contained in:
epriestley
2011-02-26 20:57:21 -08:00
parent d4bd2b0edd
commit eccc76dae6
20 changed files with 116 additions and 85 deletions

View File

@@ -370,7 +370,7 @@ class PHPMailerLite {
*/ */
private function AddAnAddress($kind, $address, $name = '') { private function AddAnAddress($kind, $address, $name = '') {
if (!preg_match('/^(to|cc|bcc|ReplyTo)$/', $kind)) { if (!preg_match('/^(to|cc|bcc|ReplyTo)$/', $kind)) {
echo 'Invalid recipient array: ' . kind; echo 'Invalid recipient array: ' . $kind;
return false; return false;
} }
$address = trim($address); $address = trim($address);

View File

@@ -261,7 +261,7 @@ function xhprof_aggregate_runs($xhprof_runs_impl, $runs,
$bad_runs = array(); $bad_runs = array();
foreach($runs as $idx => $run_id) { foreach($runs as $idx => $run_id) {
$raw_data = $xhprof_runs_impl->get_run($run_id, $source, $description); $raw_data = $xhprof_runs_impl->get_run($run_id, $source, '?');
// use the first run to derive what metrics to aggregate on. // use the first run to derive what metrics to aggregate on.
if ($idx == 0) { if ($idx == 0) {
@@ -283,7 +283,7 @@ function xhprof_aggregate_runs($xhprof_runs_impl, $runs,
} }
if ($use_script_name) { if ($use_script_name) {
$page = $description; $page = '?';
// create a fake function '__script::$page', and have and edge from // create a fake function '__script::$page', and have and edge from
// main() to '__script::$page'. We will also need edges to transfer // main() to '__script::$page'. We will also need edges to transfer
@@ -589,7 +589,7 @@ function xhprof_prune_run($raw_data, $prune_percent) {
$prune_threshold = (($main_info[$prune_metric] * $prune_percent) / 100.0); $prune_threshold = (($main_info[$prune_metric] * $prune_percent) / 100.0);
init_metrics($raw_data, null, null, false); // init_metrics($raw_data, null, null, false);
$flat_info = xhprof_compute_inclusive_times($raw_data); $flat_info = xhprof_compute_inclusive_times($raw_data);
foreach ($raw_data as $parent_child => $info) { foreach ($raw_data as $parent_child => $info) {

View File

@@ -212,12 +212,17 @@ class AphrontDefaultApplicationConfiguration
'<code>'.phutil_escape_html((string)$ex).'</code>'. '<code>'.phutil_escape_html((string)$ex).'</code>'.
'</div>'; '</div>';
if ($this->getRequest()->isAjax()) { $user = $this->getRequest()->getUser();
if (!$user) {
// If we hit an exception very early, we won't have a user.
$user = new PhabricatorUser();
}
$dialog = new AphrontDialogView(); $dialog = new AphrontDialogView();
$dialog $dialog
->setTitle('Exception!') ->setTitle('Exception!')
->setClass('aphront-exception-dialog') ->setClass('aphront-exception-dialog')
->setUser($this->getRequest()->getUser()) ->setUser($user)
->appendChild($content) ->appendChild($content)
->addCancelButton('/'); ->addCancelButton('/');
@@ -227,16 +232,6 @@ class AphrontDefaultApplicationConfiguration
return $response; return $response;
} }
$view = new PhabricatorStandardPageView();
$view->setRequest($this->getRequest());
$view->appendChild($content);
$response = new AphrontWebpageResponse();
$response->setContent($view->render());
return $response;
}
public function willSendResponse(AphrontResponse $response) { public function willSendResponse(AphrontResponse $response) {
$request = $this->getRequest(); $request = $this->getRequest();
if ($response instanceof AphrontDialogResponse) { if ($response instanceof AphrontDialogResponse) {

View File

@@ -12,6 +12,7 @@ phutil_require_module('phabricator', 'aphront/response/ajax');
phutil_require_module('phabricator', 'aphront/response/dialog'); phutil_require_module('phabricator', 'aphront/response/dialog');
phutil_require_module('phabricator', 'aphront/response/webpage'); phutil_require_module('phabricator', 'aphront/response/webpage');
phutil_require_module('phabricator', 'applications/base/controller/404'); phutil_require_module('phabricator', 'applications/base/controller/404');
phutil_require_module('phabricator', 'applications/people/storage/user');
phutil_require_module('phabricator', 'view/dialog'); phutil_require_module('phabricator', 'view/dialog');
phutil_require_module('phabricator', 'view/page/failure'); phutil_require_module('phabricator', 'view/page/failure');
phutil_require_module('phabricator', 'view/page/standard'); phutil_require_module('phabricator', 'view/page/standard');

View File

@@ -65,21 +65,30 @@ class PhabricatorOAuthLoginController extends PhabricatorAuthController {
'code' => $code, 'code' => $code,
); );
$post_data = http_build_query($query_data);
$post_length = strlen($post_data);
$stream_context = stream_context_create( $stream_context = stream_context_create(
array( array(
'http' => array( 'http' => array(
'method' => 'POST', 'method' => 'POST',
'header' => 'Content-type: application/x-www-form-urlencoded', 'header' =>
'content' => http_build_query($query_data), "Content-Type: application/x-www-form-urlencoded\r\n".
"Content-Length: {$post_length}\r\n",
'content' => $post_data,
), ),
)); ));
$stream = fopen($auth_uri, 'r', false, $stream_context); $stream = fopen($auth_uri, 'r', false, $stream_context);
$response = false;
$meta = null;
if ($stream) {
$meta = stream_get_meta_data($stream); $meta = stream_get_meta_data($stream);
$response = stream_get_contents($stream); $response = stream_get_contents($stream);
fclose($stream); fclose($stream);
}
if ($response === false) { if ($response === false) {
return $this->buildErrorResponse(new PhabricatorOAuthFailureView()); return $this->buildErrorResponse(new PhabricatorOAuthFailureView());
@@ -127,7 +136,6 @@ class PhabricatorOAuthLoginController extends PhabricatorAuthController {
$user_id); $user_id);
if ($current_user->getPHID()) { if ($current_user->getPHID()) {
if ($known_oauth) { if ($known_oauth) {
if ($known_oauth->getUserID() != $current_user->getID()) { if ($known_oauth->getUserID() != $current_user->getID()) {
$dialog = new AphrontDialogView(); $dialog = new AphrontDialogView();
@@ -285,11 +293,19 @@ class PhabricatorOAuthLoginController extends PhabricatorAuthController {
$request->setCookie('phsid', $session_key); $request->setCookie('phsid', $session_key);
return id(new AphrontRedirectResponse())->setURI('/'); return id(new AphrontRedirectResponse())->setURI('/');
} catch (AphrontQueryDuplicateKeyException $exception) { } catch (AphrontQueryDuplicateKeyException $exception) {
$key = $exception->getDuplicateKey();
if ($key == 'userName') { $same_username = id(new PhabricatorUser())->loadOneWhere(
'userName = %s',
$user->getUserName());
$same_email = id(new PhabricatorUser())->loadOneWhere(
'email = %s',
$user->getEmail());
if ($same_username) {
$e_username = 'Duplicate'; $e_username = 'Duplicate';
$errors[] = 'That username is not unique.'; $errors[] = 'That username or email is not unique.';
} else if ($key == 'email') { } else if ($same_email) {
$e_email = 'Duplicate'; $e_email = 'Duplicate';
$errors[] = 'That email is not unique.'; $errors[] = 'That email is not unique.';
} else { } else {

View File

@@ -60,8 +60,7 @@ class ConduitAPI_differential_markcommitted_Method extends ConduitAPIMethod {
$editor = new DifferentialCommentEditor( $editor = new DifferentialCommentEditor(
$revision, $revision,
$revision->getAuthorPHID(), $revision->getAuthorPHID(),
DifferentialAction::ACTION_COMMIT, DifferentialAction::ACTION_COMMIT);
$inline_comments = array());
$editor->save(); $editor->save();
$revision->setStatus(DifferentialRevisionStatus::COMMITTED); $revision->setStatus(DifferentialRevisionStatus::COMMITTED);

View File

@@ -165,7 +165,7 @@ class DifferentialRevisionListData {
$this->revisions = $rev->loadAllFromArray($data); $this->revisions = $rev->loadAllFromArray($data);
break; break;
case self::QUERY_BY_PHID: case self::QUERY_PHIDS:
$this->revisions = $this->loadAllWhere( $this->revisions = $this->loadAllWhere(
'revision.phid in (%Ls)', 'revision.phid in (%Ls)',
$this->ids); $this->ids);
@@ -233,8 +233,11 @@ class DifferentialRevisionListData {
} }
private function loadAllOpenWithCCs(array $ccphids) { private function loadAllOpenWithCCs(array $ccphids) {
$rev = new DifferentialRevision();
$revision = new DifferentialRevision(); $revision = new DifferentialRevision();
$data = queryfx_all( $data = queryfx_all(
$rev->establishConnection('r'),
'SELECT revision.* FROM %T revision 'SELECT revision.* FROM %T revision
JOIN %T relationship ON relationship.revisionID = revision.id JOIN %T relationship ON relationship.revisionID = revision.id
AND relationship.relation = %s AND relationship.relation = %s

View File

@@ -516,7 +516,7 @@ class DifferentialChangesetParser {
$changeset->getTableName().'_parse_cache', $changeset->getTableName().'_parse_cache',
$this->changesetID, $this->changesetID,
$cache); $cache);
} catch (QueryException $ex) { } catch (AphrontQueryException $ex) {
// TODO: uhoh // TODO: uhoh
} }
} }

View File

@@ -35,7 +35,7 @@ class ManiphestTaskListView extends AphrontView {
$views = array(); $views = array();
foreach ($this->tasks as $task) { foreach ($this->tasks as $task) {
$view = new ManiphestTaskSummaryView($task); $view = new ManiphestTaskSummaryView();
$view->setTask($task); $view->setTask($task);
$view->setHandles($this->handles); $view->setHandles($this->handles);
$views[] = $view->render(); $views[] = $view->render();

View File

@@ -72,7 +72,7 @@ class ManiphestTransactionListView extends AphrontView {
} }
foreach ($groups as $group) { foreach ($groups as $group) {
$view = new ManiphestTransactionDetailView($transaction); $view = new ManiphestTransactionDetailView();
$view->setTransactionGroup($group); $view->setTransactionGroup($group);
$view->setHandles($this->handles); $view->setHandles($this->handles);
$view->setMarkupEngine($this->markupEngine); $view->setMarkupEngine($this->markupEngine);

View File

@@ -69,14 +69,10 @@ class PhabricatorRepositoryCreateController extends PhabricatorController {
return id(new AphrontRedirectResponse()) return id(new AphrontRedirectResponse())
->setURI('/repository/edit/'.$repository->getID().'/'); ->setURI('/repository/edit/'.$repository->getID().'/');
} catch (PhabricatorQueryDuplicateKeyException $ex) { } catch (AphrontQueryDuplicateKeyException $ex) {
if ($ex->getDuplicateKey() == 'callsign') {
$e_callsign = 'Duplicate'; $e_callsign = 'Duplicate';
$errors[] = 'Callsign must be unique. Another repository already '. $errors[] = 'Callsign must be unique. Another repository already '.
'uses that callsign.'; 'uses that callsign.';
} else {
throw $ex;
}
} }
} }
} }

View File

@@ -36,15 +36,21 @@ abstract class AphrontDatabaseConnection {
abstract public function escapeStringForLikeClause($string); abstract public function escapeStringForLikeClause($string);
public function queryData($pattern/*, $arg, $arg, ... */) { public function queryData($pattern/*, $arg, $arg, ... */) {
if (false) {
// Workaround for the HPHP workaround: ensure we include this module
// since we really are using the function.
queryfx($this, $pattern);
}
$args = func_get_args(); $args = func_get_args();
array_unshift($args, $this); array_unshift($args, $this);
return call_user_func_array('queryfx_all', $args); return hphp_workaround_call_user_func_array('queryfx_all', $args);
} }
public function query($pattern/*, $arg, $arg, ... */) { public function query($pattern/*, $arg, $arg, ... */) {
$args = func_get_args(); $args = func_get_args();
array_unshift($args, $this); array_unshift($args, $this);
return call_user_func_array('queryfx', $args); return hphp_workaround_call_user_func_array('queryfx', $args);
} }
// TODO: Probably need to reset these when we catch a connection exception // TODO: Probably need to reset these when we catch a connection exception

View File

@@ -234,12 +234,11 @@ class AphrontMySQLDatabaseConnection extends AphrontDatabaseConnection {
case 1205: // Lock wait timeout exceeded case 1205: // Lock wait timeout exceeded
throw new AphrontQueryRecoverableException("#{$errno}: {$error}"); throw new AphrontQueryRecoverableException("#{$errno}: {$error}");
case 1062: // Duplicate Key case 1062: // Duplicate Key
$matches = null; // NOTE: In some versions of MySQL we get a key name back here, but
$key = null; // older versions just give us a key index ("key 2") so it's not
if (preg_match('/for key \'(.*)\'$/', $error, $matches)) { // portable to parse the key out of the error and attach it to the
$key = $matches[1]; // exception.
} throw new AphrontQueryDuplicateKeyException("{$errno}: {$error}");
throw new AphrontQueryDuplicateKeyException($key, "{$errno}: {$error}");
default: default:
// TODO: 1064 is syntax error, and quite terrible in production. // TODO: 1064 is syntax error, and quite terrible in production.
throw new AphrontQueryException("#{$errno}: {$error}"); throw new AphrontQueryException("#{$errno}: {$error}");

View File

@@ -21,14 +21,4 @@
*/ */
class AphrontQueryDuplicateKeyException extends AphrontQueryException { class AphrontQueryDuplicateKeyException extends AphrontQueryException {
private $duplicateKey;
public function getDuplicateKey() {
return $this->duplicateKey;
}
public function __construct($duplicate_key, $message) {
$this->duplicateKey = $duplicate_key;
parent::__construct($message);
}
} }

View File

@@ -20,9 +20,15 @@
* @group storage * @group storage
*/ */
function queryfx(AphrontDatabaseConnection $conn, $sql/*, ... */) { function queryfx(AphrontDatabaseConnection $conn, $sql/*, ... */) {
if (false) {
// Workaround for the HPHP workaround: ensure we include this module
// since we really are using the function.
qsprintf($conn, $sql);
}
$argv = func_get_args(); $argv = func_get_args();
$query = call_user_func_array('qsprintf', $argv); $query = hphp_workaround_call_user_func_array('qsprintf', $argv);
return $conn->executeRawQuery($query); $conn->executeRawQuery($query);
} }
/** /**
@@ -30,7 +36,7 @@ function queryfx(AphrontDatabaseConnection $conn, $sql/*, ... */) {
*/ */
function vqueryfx($conn, $sql, $argv) { function vqueryfx($conn, $sql, $argv) {
array_unshift($argv, $conn, $sql); array_unshift($argv, $conn, $sql);
return call_user_func_array('queryfx', $argv); hphp_workaround_call_user_func_array('queryfx', $argv);
} }
/** /**
@@ -38,8 +44,8 @@ function vqueryfx($conn, $sql, $argv) {
*/ */
function queryfx_all($conn, $sql/*, ... */) { function queryfx_all($conn, $sql/*, ... */) {
$argv = func_get_args(); $argv = func_get_args();
$ret = call_user_func_array('queryfx', $argv); hphp_workaround_call_user_func_array('queryfx', $argv);
return $conn->selectAllResults($ret); return $conn->selectAllResults();
} }
/** /**
@@ -47,7 +53,7 @@ function queryfx_all($conn, $sql/*, ... */) {
*/ */
function queryfx_one($conn, $sql/*, ... */) { function queryfx_one($conn, $sql/*, ... */) {
$argv = func_get_args(); $argv = func_get_args();
$ret = call_user_func_array('queryfx_all', $argv); $ret = hphp_workaround_call_user_func_array('queryfx_all', $argv);
if (count($ret) > 1) { if (count($ret) > 1) {
throw new AphrontQueryCountException( throw new AphrontQueryCountException(
'Query returned more than one row.'); 'Query returned more than one row.');
@@ -59,6 +65,6 @@ function queryfx_one($conn, $sql/*, ... */) {
function vqueryfx_all($conn, $sql, array $argv) { function vqueryfx_all($conn, $sql, array $argv) {
array_unshift($argv, $conn, $sql); array_unshift($argv, $conn, $sql);
$ret = call_user_func_array('queryfx', $argv); hphp_workaround_call_user_func_array('queryfx', $argv);
return $conn->selectAllResults($ret); return $conn->selectAllResults();
} }

View File

@@ -6,6 +6,7 @@
phutil_require_module('phabricator', 'infrastructure/celerity/api');
phutil_require_module('phabricator', 'view/base'); phutil_require_module('phabricator', 'view/base');

View File

@@ -152,9 +152,12 @@ class PhabricatorStandardPageView extends AphrontPageView {
$login_stuff = null; $login_stuff = null;
$request = $this->getRequest(); $request = $this->getRequest();
$user = null;
if ($request) { if ($request) {
$user = $request->getUser(); $user = $request->getUser();
if ($user->getPHID()) { // NOTE: user may not be set here if we caught an exception early
// in the execution workflow.
if ($user && $user->getPHID()) {
$login_stuff = $login_stuff =
'Logged in as '.phutil_render_tag( 'Logged in as '.phutil_render_tag(
'a', 'a',
@@ -203,6 +206,8 @@ class PhabricatorStandardPageView extends AphrontPageView {
} }
$foot_links[] = $link; $foot_links[] = $link;
} }
if ($user && $user->getPHID()) {
// This ends up very early in tab order at the top of the page and there's // This ends up very early in tab order at the top of the page and there's
// a bunch of junk up there anyway, just shove it down here. // a bunch of junk up there anyway, just shove it down here.
$foot_links[] = phabricator_render_form( $foot_links[] = phabricator_render_form(
@@ -213,6 +218,7 @@ class PhabricatorStandardPageView extends AphrontPageView {
'style' => 'display: inline', 'style' => 'display: inline',
), ),
'<button class="link">Logout</button>'); '<button class="link">Logout</button>');
}
$foot_links = implode(' &middot; ', $foot_links); $foot_links = implode(' &middot; ', $foot_links);

View File

@@ -19,6 +19,11 @@
error_reporting(E_ALL | E_STRICT); error_reporting(E_ALL | E_STRICT);
$env = getenv('PHABRICATOR_ENV'); // Apache $env = getenv('PHABRICATOR_ENV'); // Apache
if (!$env) {
if (isset($_ENV['PHABRICATOR_ENV'])) {
$env = $_ENV['PHABRICATOR_ENV'];
}
}
if (!$env) { if (!$env) {
phabricator_fatal_config_error( phabricator_fatal_config_error(
@@ -172,3 +177,11 @@ function phabricator_fatal_config_error($msg) {
die(); die();
} }
/**
* Workaround for HipHop bug, see Facebook Task #503624.
*/
function hphp_workaround_call_user_func_array($func, array $array) {
$f = new ReflectionFunction($func);
return $f->invokeArgs($array);
}