Provide a real object ("PhabricatorRepositoryPushEvent") to represent an entire push transaction
Summary: Ref T4677. Currently, we record individual actions in a push as PhabricatorRepositoryPushLogs, but tie them together only loosely with a `transactionKey`. Provide a real PushEvent object, and move some of the denormalized fields to it. This primarily just gives us more robust infrastructure for building, e.g., email about pushes, for T4677, since we can act on real PHIDs rather than passing awkward identifiers around. Test Plan: - Performed migration. - Looked at database for consistency. - Browsed/queried push logs. - Pushed a bunch of stuff. Reviewers: btrahan Reviewed By: btrahan Subscribers: epriestley Maniphest Tasks: T4677 Differential Revision: https://secure.phabricator.com/D8615
This commit is contained in:
		
							
								
								
									
										15
									
								
								resources/sql/autopatches/20140325.push.1.event.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								resources/sql/autopatches/20140325.push.1.event.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | |||||||
|  | CREATE TABLE {$NAMESPACE}_repository.repository_pushevent ( | ||||||
|  |   id INT UNSIGNED NOT NULL AUTO_INCREMENT PRIMARY KEY, | ||||||
|  |   phid VARCHAR(64) NOT NULL COLLATE utf8_bin, | ||||||
|  |   repositoryPHID  VARCHAR(64) NOT NULL COLLATE utf8_bin, | ||||||
|  |   epoch INT UNSIGNED NOT NULL, | ||||||
|  |   pusherPHID VARCHAR(64) NOT NULL COLLATE utf8_bin, | ||||||
|  |   remoteAddress INT UNSIGNED, | ||||||
|  |   remoteProtocol VARCHAR(32), | ||||||
|  |   rejectCode INT UNSIGNED NOT NULL, | ||||||
|  |   rejectDetails VARCHAR(64) COLLATE utf8_bin, | ||||||
|  |  | ||||||
|  |   UNIQUE KEY `key_phid` (phid), | ||||||
|  |   KEY `key_repository` (repositoryPHID) | ||||||
|  |  | ||||||
|  | ) ENGINE=InnoDB, COLLATE=utf8_general_ci; | ||||||
							
								
								
									
										5
									
								
								resources/sql/autopatches/20140325.push.2.eventphid.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								resources/sql/autopatches/20140325.push.2.eventphid.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | ALTER TABLE {$NAMESPACE}_repository.repository_pushlog | ||||||
|  |   ADD pushEventPHID VARCHAR(64) NOT NULL COLLATE utf8_bin AFTER epoch; | ||||||
|  |  | ||||||
|  | ALTER TABLE {$NAMESPACE}_repository.repository_pushlog | ||||||
|  |   ADD KEY `key_event` (pushEventPHID); | ||||||
							
								
								
									
										43
									
								
								resources/sql/autopatches/20140325.push.3.groups.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								resources/sql/autopatches/20140325.push.3.groups.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | $conn_w = id(new PhabricatorRepository())->establishConnection('w'); | ||||||
|  |  | ||||||
|  | echo "Adding transaction log event groups...\n"; | ||||||
|  |  | ||||||
|  | $logs = queryfx_all( | ||||||
|  |   $conn_w, | ||||||
|  |   'SELECT * FROM %T GROUP BY transactionKey ORDER BY id ASC', | ||||||
|  |   'repository_pushlog'); | ||||||
|  | foreach ($logs as $log) { | ||||||
|  |   $id = $log['id']; | ||||||
|  |   echo "Migrating log {$id}...\n"; | ||||||
|  |   if ($log['pushEventPHID']) { | ||||||
|  |     continue; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   $event_phid = id(new PhabricatorRepositoryPushEvent())->generatePHID(); | ||||||
|  |  | ||||||
|  |   queryfx( | ||||||
|  |     $conn_w, | ||||||
|  |     'INSERT INTO %T (phid, repositoryPHID, epoch, pusherPHID, remoteAddress, | ||||||
|  |       remoteProtocol, rejectCode, rejectDetails) | ||||||
|  |      VALUES (%s, %s, %d, %s, %d, %s, %d, %s)', | ||||||
|  |     'repository_pushevent', | ||||||
|  |     $event_phid, | ||||||
|  |     $log['repositoryPHID'], | ||||||
|  |     $log['epoch'], | ||||||
|  |     $log['pusherPHID'], | ||||||
|  |     $log['remoteAddress'], | ||||||
|  |     $log['remoteProtocol'], | ||||||
|  |     $log['rejectCode'], | ||||||
|  |     $log['rejectDetails']); | ||||||
|  |  | ||||||
|  |   queryfx( | ||||||
|  |     $conn_w, | ||||||
|  |     'UPDATE %T SET pushEventPHID = %s WHERE transactionKey = %s', | ||||||
|  |     'repository_pushlog', | ||||||
|  |     $event_phid, | ||||||
|  |     $log['transactionKey']); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | echo "Done.\n"; | ||||||
							
								
								
									
										14
									
								
								resources/sql/autopatches/20140325.push.4.prune.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								resources/sql/autopatches/20140325.push.4.prune.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | |||||||
|  | ALTER TABLE {$NAMESPACE}_repository.repository_pushlog | ||||||
|  |   DROP remoteAddress; | ||||||
|  |  | ||||||
|  | ALTER TABLE {$NAMESPACE}_repository.repository_pushlog | ||||||
|  |   DROP remoteProtocol; | ||||||
|  |  | ||||||
|  | ALTER TABLE {$NAMESPACE}_repository.repository_pushlog | ||||||
|  |   DROP transactionKey; | ||||||
|  |  | ||||||
|  | ALTER TABLE {$NAMESPACE}_repository.repository_pushlog | ||||||
|  |   DROP rejectCode; | ||||||
|  |  | ||||||
|  | ALTER TABLE {$NAMESPACE}_repository.repository_pushlog | ||||||
|  |   DROP rejectDetails; | ||||||
| @@ -1958,11 +1958,14 @@ phutil_register_library_map(array( | |||||||
|     'PhabricatorRepositoryPHIDTypeArcanistProject' => 'applications/repository/phid/PhabricatorRepositoryPHIDTypeArcanistProject.php', |     'PhabricatorRepositoryPHIDTypeArcanistProject' => 'applications/repository/phid/PhabricatorRepositoryPHIDTypeArcanistProject.php', | ||||||
|     'PhabricatorRepositoryPHIDTypeCommit' => 'applications/repository/phid/PhabricatorRepositoryPHIDTypeCommit.php', |     'PhabricatorRepositoryPHIDTypeCommit' => 'applications/repository/phid/PhabricatorRepositoryPHIDTypeCommit.php', | ||||||
|     'PhabricatorRepositoryPHIDTypeMirror' => 'applications/repository/phid/PhabricatorRepositoryPHIDTypeMirror.php', |     'PhabricatorRepositoryPHIDTypeMirror' => 'applications/repository/phid/PhabricatorRepositoryPHIDTypeMirror.php', | ||||||
|  |     'PhabricatorRepositoryPHIDTypePushEvent' => 'applications/repository/phid/PhabricatorRepositoryPHIDTypePushEvent.php', | ||||||
|     'PhabricatorRepositoryPHIDTypePushLog' => 'applications/repository/phid/PhabricatorRepositoryPHIDTypePushLog.php', |     'PhabricatorRepositoryPHIDTypePushLog' => 'applications/repository/phid/PhabricatorRepositoryPHIDTypePushLog.php', | ||||||
|     'PhabricatorRepositoryPHIDTypeRepository' => 'applications/repository/phid/PhabricatorRepositoryPHIDTypeRepository.php', |     'PhabricatorRepositoryPHIDTypeRepository' => 'applications/repository/phid/PhabricatorRepositoryPHIDTypeRepository.php', | ||||||
|     'PhabricatorRepositoryParsedChange' => 'applications/repository/data/PhabricatorRepositoryParsedChange.php', |     'PhabricatorRepositoryParsedChange' => 'applications/repository/data/PhabricatorRepositoryParsedChange.php', | ||||||
|     'PhabricatorRepositoryPullEngine' => 'applications/repository/engine/PhabricatorRepositoryPullEngine.php', |     'PhabricatorRepositoryPullEngine' => 'applications/repository/engine/PhabricatorRepositoryPullEngine.php', | ||||||
|     'PhabricatorRepositoryPullLocalDaemon' => 'applications/repository/daemon/PhabricatorRepositoryPullLocalDaemon.php', |     'PhabricatorRepositoryPullLocalDaemon' => 'applications/repository/daemon/PhabricatorRepositoryPullLocalDaemon.php', | ||||||
|  |     'PhabricatorRepositoryPushEvent' => 'applications/repository/storage/PhabricatorRepositoryPushEvent.php', | ||||||
|  |     'PhabricatorRepositoryPushEventQuery' => 'applications/repository/query/PhabricatorRepositoryPushEventQuery.php', | ||||||
|     'PhabricatorRepositoryPushLog' => 'applications/repository/storage/PhabricatorRepositoryPushLog.php', |     'PhabricatorRepositoryPushLog' => 'applications/repository/storage/PhabricatorRepositoryPushLog.php', | ||||||
|     'PhabricatorRepositoryPushLogQuery' => 'applications/repository/query/PhabricatorRepositoryPushLogQuery.php', |     'PhabricatorRepositoryPushLogQuery' => 'applications/repository/query/PhabricatorRepositoryPushLogQuery.php', | ||||||
|     'PhabricatorRepositoryPushLogSearchEngine' => 'applications/repository/query/PhabricatorRepositoryPushLogSearchEngine.php', |     'PhabricatorRepositoryPushLogSearchEngine' => 'applications/repository/query/PhabricatorRepositoryPushLogSearchEngine.php', | ||||||
| @@ -4792,11 +4795,18 @@ phutil_register_library_map(array( | |||||||
|     'PhabricatorRepositoryPHIDTypeArcanistProject' => 'PhabricatorPHIDType', |     'PhabricatorRepositoryPHIDTypeArcanistProject' => 'PhabricatorPHIDType', | ||||||
|     'PhabricatorRepositoryPHIDTypeCommit' => 'PhabricatorPHIDType', |     'PhabricatorRepositoryPHIDTypeCommit' => 'PhabricatorPHIDType', | ||||||
|     'PhabricatorRepositoryPHIDTypeMirror' => 'PhabricatorPHIDType', |     'PhabricatorRepositoryPHIDTypeMirror' => 'PhabricatorPHIDType', | ||||||
|  |     'PhabricatorRepositoryPHIDTypePushEvent' => 'PhabricatorPHIDType', | ||||||
|     'PhabricatorRepositoryPHIDTypePushLog' => 'PhabricatorPHIDType', |     'PhabricatorRepositoryPHIDTypePushLog' => 'PhabricatorPHIDType', | ||||||
|     'PhabricatorRepositoryPHIDTypeRepository' => 'PhabricatorPHIDType', |     'PhabricatorRepositoryPHIDTypeRepository' => 'PhabricatorPHIDType', | ||||||
|     'PhabricatorRepositoryParsedChange' => 'Phobject', |     'PhabricatorRepositoryParsedChange' => 'Phobject', | ||||||
|     'PhabricatorRepositoryPullEngine' => 'PhabricatorRepositoryEngine', |     'PhabricatorRepositoryPullEngine' => 'PhabricatorRepositoryEngine', | ||||||
|     'PhabricatorRepositoryPullLocalDaemon' => 'PhabricatorDaemon', |     'PhabricatorRepositoryPullLocalDaemon' => 'PhabricatorDaemon', | ||||||
|  |     'PhabricatorRepositoryPushEvent' => | ||||||
|  |     array( | ||||||
|  |       0 => 'PhabricatorRepositoryDAO', | ||||||
|  |       1 => 'PhabricatorPolicyInterface', | ||||||
|  |     ), | ||||||
|  |     'PhabricatorRepositoryPushEventQuery' => 'PhabricatorCursorPagedPolicyAwareQuery', | ||||||
|     'PhabricatorRepositoryPushLog' => |     'PhabricatorRepositoryPushLog' => | ||||||
|     array( |     array( | ||||||
|       0 => 'PhabricatorRepositoryDAO', |       0 => 'PhabricatorRepositoryDAO', | ||||||
|   | |||||||
| @@ -52,7 +52,7 @@ final class DiffusionPushLogListController extends DiffusionController | |||||||
|       // Reveal this if it's valid and the user can edit the repository. |       // Reveal this if it's valid and the user can edit the repository. | ||||||
|       $remote_addr = '-'; |       $remote_addr = '-'; | ||||||
|       if (isset($editable_repos[$log->getRepositoryPHID()])) { |       if (isset($editable_repos[$log->getRepositoryPHID()])) { | ||||||
|         $remote_long = $log->getRemoteAddress(); |         $remote_long = $log->getPushEvent()->getRemoteAddress(); | ||||||
|         if ($remote_long) { |         if ($remote_long) { | ||||||
|           $remote_addr = long2ip($remote_long); |           $remote_addr = long2ip($remote_long); | ||||||
|         } |         } | ||||||
| @@ -60,6 +60,7 @@ final class DiffusionPushLogListController extends DiffusionController | |||||||
|  |  | ||||||
|       $callsign = $log->getRepository()->getCallsign(); |       $callsign = $log->getRepository()->getCallsign(); | ||||||
|       $rows[] = array( |       $rows[] = array( | ||||||
|  |         $log->getPushEvent()->getID(), | ||||||
|         phutil_tag( |         phutil_tag( | ||||||
|           'a', |           'a', | ||||||
|           array( |           array( | ||||||
| @@ -68,7 +69,7 @@ final class DiffusionPushLogListController extends DiffusionController | |||||||
|           $callsign), |           $callsign), | ||||||
|         $this->getHandle($log->getPusherPHID())->renderLink(), |         $this->getHandle($log->getPusherPHID())->renderLink(), | ||||||
|         $remote_addr, |         $remote_addr, | ||||||
|         $log->getRemoteProtocol(), |         $log->getPushEvent()->getRemoteProtocol(), | ||||||
|         $log->getRefType(), |         $log->getRefType(), | ||||||
|         $log->getRefName(), |         $log->getRefName(), | ||||||
|         phutil_tag( |         phutil_tag( | ||||||
| @@ -86,7 +87,7 @@ final class DiffusionPushLogListController extends DiffusionController | |||||||
|  |  | ||||||
|         // TODO: Make these human-readable. |         // TODO: Make these human-readable. | ||||||
|         $log->getChangeFlags(), |         $log->getChangeFlags(), | ||||||
|         $log->getRejectCode(), |         $log->getPushEvent()->getRejectCode(), | ||||||
|         phabricator_datetime($log->getEpoch(), $viewer), |         phabricator_datetime($log->getEpoch(), $viewer), | ||||||
|       ); |       ); | ||||||
|     } |     } | ||||||
| @@ -94,6 +95,7 @@ final class DiffusionPushLogListController extends DiffusionController | |||||||
|     $table = id(new AphrontTableView($rows)) |     $table = id(new AphrontTableView($rows)) | ||||||
|       ->setHeaders( |       ->setHeaders( | ||||||
|         array( |         array( | ||||||
|  |           pht('Push'), | ||||||
|           pht('Repository'), |           pht('Repository'), | ||||||
|           pht('Pusher'), |           pht('Pusher'), | ||||||
|           pht('From'), |           pht('From'), | ||||||
| @@ -113,6 +115,7 @@ final class DiffusionPushLogListController extends DiffusionController | |||||||
|           '', |           '', | ||||||
|           '', |           '', | ||||||
|           '', |           '', | ||||||
|  |           '', | ||||||
|           'wide', |           'wide', | ||||||
|           'n', |           'n', | ||||||
|           'n', |           'n', | ||||||
|   | |||||||
| @@ -30,6 +30,8 @@ final class DiffusionCommitHookEngine extends Phobject { | |||||||
|   private $gitCommits = array(); |   private $gitCommits = array(); | ||||||
|  |  | ||||||
|   private $heraldViewerProjects; |   private $heraldViewerProjects; | ||||||
|  |   private $rejectCode = PhabricatorRepositoryPushLog::REJECT_BROKEN; | ||||||
|  |   private $rejectDetails; | ||||||
|  |  | ||||||
|  |  | ||||||
| /* -(  Config  )------------------------------------------------------------- */ | /* -(  Config  )------------------------------------------------------------- */ | ||||||
| @@ -62,14 +64,6 @@ final class DiffusionCommitHookEngine extends Phobject { | |||||||
|     return $remote_address; |     return $remote_address; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private function getTransactionKey() { |  | ||||||
|     if (!$this->transactionKey) { |  | ||||||
|       $entropy = Filesystem::readRandomBytes(64); |  | ||||||
|       $this->transactionKey = PhabricatorHash::digestForIndex($entropy); |  | ||||||
|     } |  | ||||||
|     return $this->transactionKey; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   public function setSubversionTransactionInfo($transaction, $repository) { |   public function setSubversionTransactionInfo($transaction, $repository) { | ||||||
|     $this->subversionTransaction = $transaction; |     $this->subversionTransaction = $transaction; | ||||||
|     $this->subversionRepository = $repository; |     $this->subversionRepository = $repository; | ||||||
| @@ -137,10 +131,7 @@ final class DiffusionCommitHookEngine extends Phobject { | |||||||
|       } catch (DiffusionCommitHookRejectException $ex) { |       } catch (DiffusionCommitHookRejectException $ex) { | ||||||
|         // If we're rejecting dangerous changes, flag everything that we've |         // If we're rejecting dangerous changes, flag everything that we've | ||||||
|         // seen as rejected so it's clear that none of it was accepted. |         // seen as rejected so it's clear that none of it was accepted. | ||||||
|         foreach ($all_updates as $update) { |         $this->rejectCode = PhabricatorRepositoryPushLog::REJECT_DANGEROUS; | ||||||
|           $update->setRejectCode( |  | ||||||
|             PhabricatorRepositoryPushLog::REJECT_DANGEROUS); |  | ||||||
|         } |  | ||||||
|         throw $ex; |         throw $ex; | ||||||
|       } |       } | ||||||
|  |  | ||||||
| @@ -156,9 +147,7 @@ final class DiffusionCommitHookEngine extends Phobject { | |||||||
|  |  | ||||||
|       // If we make it this far, we're accepting these changes. Mark all the |       // If we make it this far, we're accepting these changes. Mark all the | ||||||
|       // logs as accepted. |       // logs as accepted. | ||||||
|       foreach ($all_updates as $update) { |       $this->rejectCode = PhabricatorRepositoryPushLog::REJECT_ACCEPT; | ||||||
|         $update->setRejectCode(PhabricatorRepositoryPushLog::REJECT_ACCEPT); |  | ||||||
|       } |  | ||||||
|     } catch (Exception $ex) { |     } catch (Exception $ex) { | ||||||
|       // We'll throw this again in a minute, but we want to save all the logs |       // We'll throw this again in a minute, but we want to save all the logs | ||||||
|       // first. |       // first. | ||||||
| @@ -166,9 +155,18 @@ final class DiffusionCommitHookEngine extends Phobject { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     // Save all the logs no matter what the outcome was. |     // Save all the logs no matter what the outcome was. | ||||||
|     foreach ($all_updates as $update) { |     $event = $this->newPushEvent(); | ||||||
|       $update->save(); |  | ||||||
|     } |     $event->setRejectCode($this->rejectCode); | ||||||
|  |     $event->setRejectDetails($this->rejectDetails); | ||||||
|  |  | ||||||
|  |     $event->openTransaction(); | ||||||
|  |       $event->save(); | ||||||
|  |       foreach ($all_updates as $update) { | ||||||
|  |         $update->setPushEventPHID($event->getPHID()); | ||||||
|  |         $update->save(); | ||||||
|  |       } | ||||||
|  |     $event->saveTransaction(); | ||||||
|  |  | ||||||
|     if ($caught) { |     if ($caught) { | ||||||
|       throw $caught; |       throw $caught; | ||||||
| @@ -296,10 +294,8 @@ final class DiffusionCommitHookEngine extends Phobject { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     if ($blocking_effect) { |     if ($blocking_effect) { | ||||||
|       foreach ($all_updates as $update) { |       $this->rejectCode = PhabricatorRepositoryPushLog::REJECT_HERALD; | ||||||
|         $update->setRejectCode(PhabricatorRepositoryPushLog::REJECT_HERALD); |       $this->rejectDetails = $blocking_effect->getRulePHID(); | ||||||
|         $update->setRejectDetails($blocking_effect->getRulePHID()); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       $message = $blocking_effect->getTarget(); |       $message = $blocking_effect->getTarget(); | ||||||
|       if (!strlen($message)) { |       if (!strlen($message)) { | ||||||
| @@ -596,12 +592,8 @@ final class DiffusionCommitHookEngine extends Phobject { | |||||||
|           continue; |           continue; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         // Mark everything as rejected by this hook. |         $this->rejectCode = PhabricatorRepositoryPushLog::REJECT_EXTERNAL; | ||||||
|         foreach ($updates as $update) { |         $this->rejectDetails = basename($hook); | ||||||
|           $update->setRejectCode( |  | ||||||
|             PhabricatorRepositoryPushLog::REJECT_EXTERNAL); |  | ||||||
|           $update->setRejectDetails(basename($hook)); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         throw new DiffusionCommitHookRejectException( |         throw new DiffusionCommitHookRejectException( | ||||||
|           pht( |           pht( | ||||||
| @@ -983,24 +975,23 @@ final class DiffusionCommitHookEngine extends Phobject { | |||||||
|  |  | ||||||
|  |  | ||||||
|   private function newPushLog() { |   private function newPushLog() { | ||||||
|     // NOTE: By default, we create these with REJECT_BROKEN as the reject |  | ||||||
|     // code. This indicates a broken hook, and covers the case where we |  | ||||||
|     // encounter some unexpected exception and consequently reject the changes. |  | ||||||
|  |  | ||||||
|     // NOTE: We generate PHIDs up front so the Herald transcripts can pick them |     // NOTE: We generate PHIDs up front so the Herald transcripts can pick them | ||||||
|     // up. |     // up. | ||||||
|     $phid = id(new PhabricatorRepositoryPushLog())->generatePHID(); |     $phid = id(new PhabricatorRepositoryPushLog())->generatePHID(); | ||||||
|  |  | ||||||
|     return PhabricatorRepositoryPushLog::initializeNewLog($this->getViewer()) |     return PhabricatorRepositoryPushLog::initializeNewLog($this->getViewer()) | ||||||
|       ->setPHID($phid) |       ->setPHID($phid) | ||||||
|       ->attachRepository($this->getRepository()) |  | ||||||
|       ->setRepositoryPHID($this->getRepository()->getPHID()) |       ->setRepositoryPHID($this->getRepository()->getPHID()) | ||||||
|       ->setEpoch(time()) |       ->setEpoch(time()); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private function newPushEvent() { | ||||||
|  |     $viewer = $this->getViewer(); | ||||||
|  |     return PhabricatorRepositoryPushEvent::initializeNewEvent($viewer) | ||||||
|  |       ->setRepositoryPHID($this->getRepository()->getPHID()) | ||||||
|       ->setRemoteAddress($this->getRemoteAddressForLog()) |       ->setRemoteAddress($this->getRemoteAddressForLog()) | ||||||
|       ->setRemoteProtocol($this->getRemoteProtocol()) |       ->setRemoteProtocol($this->getRemoteProtocol()) | ||||||
|       ->setTransactionKey($this->getTransactionKey()) |       ->setEpoch(time()); | ||||||
|       ->setRejectCode(PhabricatorRepositoryPushLog::REJECT_BROKEN) |  | ||||||
|       ->setRejectDetails(null); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public function loadChangesetsForCommit($identifier) { |   public function loadChangesetsForCommit($identifier) { | ||||||
|   | |||||||
| @@ -0,0 +1,40 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | final class PhabricatorRepositoryPHIDTypePushEvent | ||||||
|  |   extends PhabricatorPHIDType { | ||||||
|  |  | ||||||
|  |   const TYPECONST = 'PSHE'; | ||||||
|  |  | ||||||
|  |   public function getTypeConstant() { | ||||||
|  |     return self::TYPECONST; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function getTypeName() { | ||||||
|  |     return pht('Push Event'); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function newObject() { | ||||||
|  |     return new PhabricatorRepositoryPushEvent(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   protected function buildQueryForObjects( | ||||||
|  |     PhabricatorObjectQuery $query, | ||||||
|  |     array $phids) { | ||||||
|  |  | ||||||
|  |     return id(new PhabricatorRepositoryPushEventQuery()) | ||||||
|  |       ->withPHIDs($phids); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function loadHandles( | ||||||
|  |     PhabricatorHandleQuery $query, | ||||||
|  |     array $handles, | ||||||
|  |     array $objects) { | ||||||
|  |  | ||||||
|  |     foreach ($handles as $phid => $handle) { | ||||||
|  |       $event = $objects[$phid]; | ||||||
|  |  | ||||||
|  |       $handle->setName(pht('Push Event %d', $event->getID())); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,108 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | final class PhabricatorRepositoryPushEventQuery | ||||||
|  |   extends PhabricatorCursorPagedPolicyAwareQuery { | ||||||
|  |  | ||||||
|  |   private $ids; | ||||||
|  |   private $phids; | ||||||
|  |   private $repositoryPHIDs; | ||||||
|  |   private $pusherPHIDs; | ||||||
|  |  | ||||||
|  |   public function withIDs(array $ids) { | ||||||
|  |     $this->ids = $ids; | ||||||
|  |     return $this; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function withPHIDs(array $phids) { | ||||||
|  |     $this->phids = $phids; | ||||||
|  |     return $this; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function withRepositoryPHIDs(array $repository_phids) { | ||||||
|  |     $this->repositoryPHIDs = $repository_phids; | ||||||
|  |     return $this; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function withPusherPHIDs(array $pusher_phids) { | ||||||
|  |     $this->pusherPHIDs = $pusher_phids; | ||||||
|  |     return $this; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   protected function loadPage() { | ||||||
|  |     $table = new PhabricatorRepositoryPushEvent(); | ||||||
|  |     $conn_r = $table->establishConnection('r'); | ||||||
|  |  | ||||||
|  |     $data = queryfx_all( | ||||||
|  |       $conn_r, | ||||||
|  |       'SELECT * FROM %T %Q %Q %Q', | ||||||
|  |       $table->getTableName(), | ||||||
|  |       $this->buildWhereClause($conn_r), | ||||||
|  |       $this->buildOrderClause($conn_r), | ||||||
|  |       $this->buildLimitClause($conn_r)); | ||||||
|  |  | ||||||
|  |     return $table->loadAllFromArray($data); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function willFilterPage(array $events) { | ||||||
|  |     $repository_phids = mpull($events, 'getRepositoryPHID'); | ||||||
|  |     $repositories = id(new PhabricatorRepositoryQuery()) | ||||||
|  |       ->setViewer($this->getViewer()) | ||||||
|  |       ->withPHIDs($repository_phids) | ||||||
|  |       ->execute(); | ||||||
|  |     $repositories = mpull($repositories, null, 'getPHID'); | ||||||
|  |  | ||||||
|  |     foreach ($events as $key => $event) { | ||||||
|  |       $phid = $event->getRepositoryPHID(); | ||||||
|  |       if (empty($repositories[$phid])) { | ||||||
|  |         unset($events[$key]); | ||||||
|  |         continue; | ||||||
|  |       } | ||||||
|  |       $event->attachRepository($repositories[$phid]); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return $events; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   private function buildWhereClause(AphrontDatabaseConnection $conn_r) { | ||||||
|  |     $where = array(); | ||||||
|  |  | ||||||
|  |     if ($this->ids) { | ||||||
|  |       $where[] = qsprintf( | ||||||
|  |         $conn_r, | ||||||
|  |         'id IN (%Ld)', | ||||||
|  |         $this->ids); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if ($this->phids) { | ||||||
|  |       $where[] = qsprintf( | ||||||
|  |         $conn_r, | ||||||
|  |         'phid IN (%Ls)', | ||||||
|  |         $this->phids); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if ($this->repositoryPHIDs) { | ||||||
|  |       $where[] = qsprintf( | ||||||
|  |         $conn_r, | ||||||
|  |         'repositoryPHID IN (%Ls)', | ||||||
|  |         $this->repositoryPHIDs); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if ($this->pusherPHIDs) { | ||||||
|  |       $where[] = qsprintf( | ||||||
|  |         $conn_r, | ||||||
|  |         'pusherPHID in (%Ls)', | ||||||
|  |         $this->pusherPHIDs); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     $where[] = $this->buildPagingClause($conn_r); | ||||||
|  |  | ||||||
|  |     return $this->formatWhereClause($where); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   public function getQueryApplicationClass() { | ||||||
|  |     return 'PhabricatorApplicationDiffusion'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -56,24 +56,20 @@ final class PhabricatorRepositoryPushLogQuery | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   public function willFilterPage(array $logs) { |   public function willFilterPage(array $logs) { | ||||||
|     $repository_phids = mpull($logs, 'getRepositoryPHID'); |     $event_phids = mpull($logs, 'getPushEventPHID'); | ||||||
|     if ($repository_phids) { |     $events = id(new PhabricatorRepositoryPushEventQuery()) | ||||||
|       $repositories = id(new PhabricatorRepositoryQuery()) |       ->setViewer($this->getViewer()) | ||||||
|         ->setViewer($this->getViewer()) |       ->withPHIDs($event_phids) | ||||||
|         ->withPHIDs($repository_phids) |       ->execute(); | ||||||
|         ->execute(); |     $events = mpull($events, null, 'getPHID'); | ||||||
|       $repositories = mpull($repositories, null, 'getPHID'); |  | ||||||
|     } else { |  | ||||||
|       $repositories = array(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     foreach ($logs as $key => $log) { |     foreach ($logs as $key => $log) { | ||||||
|       $phid = $log->getRepositoryPHID(); |       $event = idx($events, $log->getPushEventPHID()); | ||||||
|       if (empty($repositories[$phid])) { |       if (!$event) { | ||||||
|         unset($logs[$key]); |         unset($logs[$key]); | ||||||
|         continue; |         continue; | ||||||
|       } |       } | ||||||
|       $log->attachRepository($repositories[$phid]); |       $log->attachPushEvent($event); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return $logs; |     return $logs; | ||||||
|   | |||||||
| @@ -0,0 +1,71 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Groups a set of push logs corresponding to changes which were all pushed in | ||||||
|  |  * the same transaction. | ||||||
|  |  */ | ||||||
|  | final class PhabricatorRepositoryPushEvent | ||||||
|  |   extends PhabricatorRepositoryDAO | ||||||
|  |   implements PhabricatorPolicyInterface { | ||||||
|  |  | ||||||
|  |   protected $repositoryPHID; | ||||||
|  |   protected $epoch; | ||||||
|  |   protected $pusherPHID; | ||||||
|  |   protected $remoteAddress; | ||||||
|  |   protected $remoteProtocol; | ||||||
|  |   protected $rejectCode; | ||||||
|  |   protected $rejectDetails; | ||||||
|  |  | ||||||
|  |   private $repository = self::ATTACHABLE; | ||||||
|  |  | ||||||
|  |   public static function initializeNewEvent(PhabricatorUser $viewer) { | ||||||
|  |     return id(new PhabricatorRepositoryPushEvent()) | ||||||
|  |       ->setPusherPHID($viewer->getPHID()); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function getConfiguration() { | ||||||
|  |     return array( | ||||||
|  |       self::CONFIG_AUX_PHID => true, | ||||||
|  |       self::CONFIG_TIMESTAMPS => false, | ||||||
|  |     ) + parent::getConfiguration(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function generatePHID() { | ||||||
|  |     return PhabricatorPHID::generateNewPHID( | ||||||
|  |       PhabricatorRepositoryPHIDTypePushEvent::TYPECONST); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function attachRepository(PhabricatorRepository $repository) { | ||||||
|  |     $this->repository = $repository; | ||||||
|  |     return $this; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function getRepository() { | ||||||
|  |     return $this->assertAttached($this->repository); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | /* -(  PhabricatorPolicyInterface  )----------------------------------------- */ | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   public function getCapabilities() { | ||||||
|  |     return array( | ||||||
|  |       PhabricatorPolicyCapability::CAN_VIEW, | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function getPolicy($capability) { | ||||||
|  |     return $this->getRepository()->getPolicy($capability); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function hasAutomaticCapability($capability, PhabricatorUser $viewer) { | ||||||
|  |     return $this->getRepository()->hasAutomaticCapability($capability, $viewer); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function describeAutomaticCapability($capability) { | ||||||
|  |     return pht( | ||||||
|  |       "A repository's push events are visible to users who can see the ". | ||||||
|  |       "repository."); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -33,9 +33,7 @@ final class PhabricatorRepositoryPushLog | |||||||
|   protected $repositoryPHID; |   protected $repositoryPHID; | ||||||
|   protected $epoch; |   protected $epoch; | ||||||
|   protected $pusherPHID; |   protected $pusherPHID; | ||||||
|   protected $remoteAddress; |   protected $pushEventPHID; | ||||||
|   protected $remoteProtocol; |  | ||||||
|   protected $transactionKey; |  | ||||||
|   protected $refType; |   protected $refType; | ||||||
|   protected $refNameHash; |   protected $refNameHash; | ||||||
|   protected $refNameRaw; |   protected $refNameRaw; | ||||||
| @@ -44,11 +42,9 @@ final class PhabricatorRepositoryPushLog | |||||||
|   protected $refNew; |   protected $refNew; | ||||||
|   protected $mergeBase; |   protected $mergeBase; | ||||||
|   protected $changeFlags; |   protected $changeFlags; | ||||||
|   protected $rejectCode; |  | ||||||
|   protected $rejectDetails; |  | ||||||
|  |  | ||||||
|   private $dangerousChangeDescription = self::ATTACHABLE; |   private $dangerousChangeDescription = self::ATTACHABLE; | ||||||
|   private $repository = self::ATTACHABLE; |   private $pushEvent = self::ATTACHABLE; | ||||||
|  |  | ||||||
|   public static function initializeNewLog(PhabricatorUser $viewer) { |   public static function initializeNewLog(PhabricatorUser $viewer) { | ||||||
|     return id(new PhabricatorRepositoryPushLog()) |     return id(new PhabricatorRepositoryPushLog()) | ||||||
| @@ -70,13 +66,17 @@ final class PhabricatorRepositoryPushLog | |||||||
|       PhabricatorRepositoryPHIDTypePushLog::TYPECONST); |       PhabricatorRepositoryPHIDTypePushLog::TYPECONST); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public function attachRepository(PhabricatorRepository $repository) { |   public function getRepository() { | ||||||
|     $this->repository = $repository; |     return $this->getPushEvent()->getRepository(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function attachPushEvent(PhabricatorRepositoryPushEvent $push_event) { | ||||||
|  |     $this->pushEvent = $push_event; | ||||||
|     return $this; |     return $this; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public function getRepository() { |   public function getPushEvent() { | ||||||
|     return $this->assertAttached($this->repository); |     return $this->assertAttached($this->pushEvent); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public function getRefName() { |   public function getRefName() { | ||||||
| @@ -131,11 +131,11 @@ final class PhabricatorRepositoryPushLog | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   public function getPolicy($capability) { |   public function getPolicy($capability) { | ||||||
|     return $this->getRepository()->getPolicy($capability); |     return $this->getPushEvent()->getPolicy($capability); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public function hasAutomaticCapability($capability, PhabricatorUser $viewer) { |   public function hasAutomaticCapability($capability, PhabricatorUser $viewer) { | ||||||
|     return $this->getRepository()->hasAutomaticCapability($capability, $viewer); |     return $this->getPushEvent()->hasAutomaticCapability($capability, $viewer); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public function describeAutomaticCapability($capability) { |   public function describeAutomaticCapability($capability) { | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 epriestley
					epriestley