Make the repository "Filesize Limit" and "Clone/Fetch Timeout" configurable in the UI
Summary: Depends on D19830. Ref T13216. See PHI908. See PHI750. See PHI885. Allow users to configure a filesize limit, and allow them to adjust the clone/fetch timeout.
Test Plan:
{F6021356}
  - Configured a filesize limit and pushed, hit it. Made the limit larger and pushed, change went through.
Reviewers: amckinley
Reviewed By: amckinley
Subscribers: yelirekim, PHID-OPKG-gm6ozazyms6q6i22gyam
Maniphest Tasks: T13216
Differential Revision: https://secure.phabricator.com/D19831
			
			
This commit is contained in:
		| @@ -950,6 +950,7 @@ phutil_register_library_map(array( | |||||||
|     'DiffusionRepositoryHistoryManagementPanel' => 'applications/diffusion/management/DiffusionRepositoryHistoryManagementPanel.php', |     'DiffusionRepositoryHistoryManagementPanel' => 'applications/diffusion/management/DiffusionRepositoryHistoryManagementPanel.php', | ||||||
|     'DiffusionRepositoryIdentityEditor' => 'applications/diffusion/editor/DiffusionRepositoryIdentityEditor.php', |     'DiffusionRepositoryIdentityEditor' => 'applications/diffusion/editor/DiffusionRepositoryIdentityEditor.php', | ||||||
|     'DiffusionRepositoryIdentitySearchEngine' => 'applications/diffusion/query/DiffusionRepositoryIdentitySearchEngine.php', |     'DiffusionRepositoryIdentitySearchEngine' => 'applications/diffusion/query/DiffusionRepositoryIdentitySearchEngine.php', | ||||||
|  |     'DiffusionRepositoryLimitsManagementPanel' => 'applications/diffusion/management/DiffusionRepositoryLimitsManagementPanel.php', | ||||||
|     'DiffusionRepositoryListController' => 'applications/diffusion/controller/DiffusionRepositoryListController.php', |     'DiffusionRepositoryListController' => 'applications/diffusion/controller/DiffusionRepositoryListController.php', | ||||||
|     'DiffusionRepositoryManageController' => 'applications/diffusion/controller/DiffusionRepositoryManageController.php', |     'DiffusionRepositoryManageController' => 'applications/diffusion/controller/DiffusionRepositoryManageController.php', | ||||||
|     'DiffusionRepositoryManagePanelsController' => 'applications/diffusion/controller/DiffusionRepositoryManagePanelsController.php', |     'DiffusionRepositoryManagePanelsController' => 'applications/diffusion/controller/DiffusionRepositoryManagePanelsController.php', | ||||||
| @@ -4104,6 +4105,7 @@ phutil_register_library_map(array( | |||||||
|     'PhabricatorRepositoryCommitRef' => 'applications/repository/engine/PhabricatorRepositoryCommitRef.php', |     'PhabricatorRepositoryCommitRef' => 'applications/repository/engine/PhabricatorRepositoryCommitRef.php', | ||||||
|     'PhabricatorRepositoryCommitTestCase' => 'applications/repository/storage/__tests__/PhabricatorRepositoryCommitTestCase.php', |     'PhabricatorRepositoryCommitTestCase' => 'applications/repository/storage/__tests__/PhabricatorRepositoryCommitTestCase.php', | ||||||
|     'PhabricatorRepositoryConfigOptions' => 'applications/repository/config/PhabricatorRepositoryConfigOptions.php', |     'PhabricatorRepositoryConfigOptions' => 'applications/repository/config/PhabricatorRepositoryConfigOptions.php', | ||||||
|  |     'PhabricatorRepositoryCopyTimeLimitTransaction' => 'applications/repository/xaction/PhabricatorRepositoryCopyTimeLimitTransaction.php', | ||||||
|     'PhabricatorRepositoryDAO' => 'applications/repository/storage/PhabricatorRepositoryDAO.php', |     'PhabricatorRepositoryDAO' => 'applications/repository/storage/PhabricatorRepositoryDAO.php', | ||||||
|     'PhabricatorRepositoryDangerousTransaction' => 'applications/repository/xaction/PhabricatorRepositoryDangerousTransaction.php', |     'PhabricatorRepositoryDangerousTransaction' => 'applications/repository/xaction/PhabricatorRepositoryDangerousTransaction.php', | ||||||
|     'PhabricatorRepositoryDefaultBranchTransaction' => 'applications/repository/xaction/PhabricatorRepositoryDefaultBranchTransaction.php', |     'PhabricatorRepositoryDefaultBranchTransaction' => 'applications/repository/xaction/PhabricatorRepositoryDefaultBranchTransaction.php', | ||||||
| @@ -4115,6 +4117,7 @@ phutil_register_library_map(array( | |||||||
|     'PhabricatorRepositoryEngine' => 'applications/repository/engine/PhabricatorRepositoryEngine.php', |     'PhabricatorRepositoryEngine' => 'applications/repository/engine/PhabricatorRepositoryEngine.php', | ||||||
|     'PhabricatorRepositoryEnormousTransaction' => 'applications/repository/xaction/PhabricatorRepositoryEnormousTransaction.php', |     'PhabricatorRepositoryEnormousTransaction' => 'applications/repository/xaction/PhabricatorRepositoryEnormousTransaction.php', | ||||||
|     'PhabricatorRepositoryFerretEngine' => 'applications/repository/search/PhabricatorRepositoryFerretEngine.php', |     'PhabricatorRepositoryFerretEngine' => 'applications/repository/search/PhabricatorRepositoryFerretEngine.php', | ||||||
|  |     'PhabricatorRepositoryFilesizeLimitTransaction' => 'applications/repository/xaction/PhabricatorRepositoryFilesizeLimitTransaction.php', | ||||||
|     'PhabricatorRepositoryFulltextEngine' => 'applications/repository/search/PhabricatorRepositoryFulltextEngine.php', |     'PhabricatorRepositoryFulltextEngine' => 'applications/repository/search/PhabricatorRepositoryFulltextEngine.php', | ||||||
|     'PhabricatorRepositoryGitCommitChangeParserWorker' => 'applications/repository/worker/commitchangeparser/PhabricatorRepositoryGitCommitChangeParserWorker.php', |     'PhabricatorRepositoryGitCommitChangeParserWorker' => 'applications/repository/worker/commitchangeparser/PhabricatorRepositoryGitCommitChangeParserWorker.php', | ||||||
|     'PhabricatorRepositoryGitCommitMessageParserWorker' => 'applications/repository/worker/commitmessageparser/PhabricatorRepositoryGitCommitMessageParserWorker.php', |     'PhabricatorRepositoryGitCommitMessageParserWorker' => 'applications/repository/worker/commitmessageparser/PhabricatorRepositoryGitCommitMessageParserWorker.php', | ||||||
| @@ -6361,6 +6364,7 @@ phutil_register_library_map(array( | |||||||
|     'DiffusionRepositoryHistoryManagementPanel' => 'DiffusionRepositoryManagementPanel', |     'DiffusionRepositoryHistoryManagementPanel' => 'DiffusionRepositoryManagementPanel', | ||||||
|     'DiffusionRepositoryIdentityEditor' => 'PhabricatorApplicationTransactionEditor', |     'DiffusionRepositoryIdentityEditor' => 'PhabricatorApplicationTransactionEditor', | ||||||
|     'DiffusionRepositoryIdentitySearchEngine' => 'PhabricatorApplicationSearchEngine', |     'DiffusionRepositoryIdentitySearchEngine' => 'PhabricatorApplicationSearchEngine', | ||||||
|  |     'DiffusionRepositoryLimitsManagementPanel' => 'DiffusionRepositoryManagementPanel', | ||||||
|     'DiffusionRepositoryListController' => 'DiffusionController', |     'DiffusionRepositoryListController' => 'DiffusionController', | ||||||
|     'DiffusionRepositoryManageController' => 'DiffusionController', |     'DiffusionRepositoryManageController' => 'DiffusionController', | ||||||
|     'DiffusionRepositoryManagePanelsController' => 'DiffusionRepositoryManageController', |     'DiffusionRepositoryManagePanelsController' => 'DiffusionRepositoryManageController', | ||||||
| @@ -10070,6 +10074,7 @@ phutil_register_library_map(array( | |||||||
|     'PhabricatorRepositoryCommitRef' => 'Phobject', |     'PhabricatorRepositoryCommitRef' => 'Phobject', | ||||||
|     'PhabricatorRepositoryCommitTestCase' => 'PhabricatorTestCase', |     'PhabricatorRepositoryCommitTestCase' => 'PhabricatorTestCase', | ||||||
|     'PhabricatorRepositoryConfigOptions' => 'PhabricatorApplicationConfigOptions', |     'PhabricatorRepositoryConfigOptions' => 'PhabricatorApplicationConfigOptions', | ||||||
|  |     'PhabricatorRepositoryCopyTimeLimitTransaction' => 'PhabricatorRepositoryTransactionType', | ||||||
|     'PhabricatorRepositoryDAO' => 'PhabricatorLiskDAO', |     'PhabricatorRepositoryDAO' => 'PhabricatorLiskDAO', | ||||||
|     'PhabricatorRepositoryDangerousTransaction' => 'PhabricatorRepositoryTransactionType', |     'PhabricatorRepositoryDangerousTransaction' => 'PhabricatorRepositoryTransactionType', | ||||||
|     'PhabricatorRepositoryDefaultBranchTransaction' => 'PhabricatorRepositoryTransactionType', |     'PhabricatorRepositoryDefaultBranchTransaction' => 'PhabricatorRepositoryTransactionType', | ||||||
| @@ -10081,6 +10086,7 @@ phutil_register_library_map(array( | |||||||
|     'PhabricatorRepositoryEngine' => 'Phobject', |     'PhabricatorRepositoryEngine' => 'Phobject', | ||||||
|     'PhabricatorRepositoryEnormousTransaction' => 'PhabricatorRepositoryTransactionType', |     'PhabricatorRepositoryEnormousTransaction' => 'PhabricatorRepositoryTransactionType', | ||||||
|     'PhabricatorRepositoryFerretEngine' => 'PhabricatorFerretEngine', |     'PhabricatorRepositoryFerretEngine' => 'PhabricatorFerretEngine', | ||||||
|  |     'PhabricatorRepositoryFilesizeLimitTransaction' => 'PhabricatorRepositoryTransactionType', | ||||||
|     'PhabricatorRepositoryFulltextEngine' => 'PhabricatorFulltextEngine', |     'PhabricatorRepositoryFulltextEngine' => 'PhabricatorFulltextEngine', | ||||||
|     'PhabricatorRepositoryGitCommitChangeParserWorker' => 'PhabricatorRepositoryCommitChangeParserWorker', |     'PhabricatorRepositoryGitCommitChangeParserWorker' => 'PhabricatorRepositoryCommitChangeParserWorker', | ||||||
|     'PhabricatorRepositoryGitCommitMessageParserWorker' => 'PhabricatorRepositoryCommitMessageParserWorker', |     'PhabricatorRepositoryGitCommitMessageParserWorker' => 'PhabricatorRepositoryCommitMessageParserWorker', | ||||||
|   | |||||||
| @@ -472,6 +472,25 @@ final class DiffusionRepositoryEditEngine | |||||||
|           pht('Change the push policy of the repository.')) |           pht('Change the push policy of the repository.')) | ||||||
|         ->setConduitTypeDescription(pht('New policy PHID or constant.')) |         ->setConduitTypeDescription(pht('New policy PHID or constant.')) | ||||||
|         ->setValue($object->getPolicy(DiffusionPushCapability::CAPABILITY)), |         ->setValue($object->getPolicy(DiffusionPushCapability::CAPABILITY)), | ||||||
|  |       id(new PhabricatorTextEditField()) | ||||||
|  |         ->setKey('filesizeLimit') | ||||||
|  |         ->setLabel(pht('Filesize Limit')) | ||||||
|  |         ->setTransactionType( | ||||||
|  |           PhabricatorRepositoryFilesizeLimitTransaction::TRANSACTIONTYPE) | ||||||
|  |         ->setDescription(pht('Maximum permitted file size.')) | ||||||
|  |         ->setConduitDescription(pht('Change the filesize limit.')) | ||||||
|  |         ->setConduitTypeDescription(pht('New repository filesize limit.')) | ||||||
|  |         ->setValue($object->getFilesizeLimit()), | ||||||
|  |       id(new PhabricatorTextEditField()) | ||||||
|  |         ->setKey('copyTimeLimit') | ||||||
|  |         ->setLabel(pht('Clone/Fetch Timeout')) | ||||||
|  |         ->setTransactionType( | ||||||
|  |           PhabricatorRepositoryCopyTimeLimitTransaction::TRANSACTIONTYPE) | ||||||
|  |         ->setDescription( | ||||||
|  |           pht('Maximum permitted duration of internal clone/fetch.')) | ||||||
|  |         ->setConduitDescription(pht('Change the copy time limit.')) | ||||||
|  |         ->setConduitTypeDescription(pht('New repository copy time limit.')) | ||||||
|  |         ->setValue($object->getCopyTimeLimit()), | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1268,9 +1268,7 @@ final class DiffusionCommitHookEngine extends Phobject { | |||||||
|   private function rejectOversizedFiles(array $content_updates) { |   private function rejectOversizedFiles(array $content_updates) { | ||||||
|     $repository = $this->getRepository(); |     $repository = $this->getRepository(); | ||||||
|  |  | ||||||
|     // TODO: Allow repositories to be configured for a maximum filesize. |     $limit = $repository->getFilesizeLimit(); | ||||||
|     $limit = 0; |  | ||||||
|  |  | ||||||
|     if (!$limit) { |     if (!$limit) { | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -0,0 +1,101 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | final class DiffusionRepositoryLimitsManagementPanel | ||||||
|  |   extends DiffusionRepositoryManagementPanel { | ||||||
|  |  | ||||||
|  |   const PANELKEY = 'limits'; | ||||||
|  |  | ||||||
|  |   public function getManagementPanelLabel() { | ||||||
|  |     return pht('Limits'); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function getManagementPanelOrder() { | ||||||
|  |     return 700; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function shouldEnableForRepository( | ||||||
|  |     PhabricatorRepository $repository) { | ||||||
|  |     return $repository->isGit(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function getManagementPanelIcon() { | ||||||
|  |     $repository = $this->getRepository(); | ||||||
|  |  | ||||||
|  |     $any_limit = false; | ||||||
|  |  | ||||||
|  |     if ($repository->getFilesizeLimit()) { | ||||||
|  |       $any_limit = true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if ($any_limit) { | ||||||
|  |       return 'fa-signal'; | ||||||
|  |     } else { | ||||||
|  |       return 'fa-signal grey'; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   protected function getEditEngineFieldKeys() { | ||||||
|  |     return array( | ||||||
|  |       'filesizeLimit', | ||||||
|  |       'copyTimeLimit', | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function buildManagementPanelCurtain() { | ||||||
|  |     $repository = $this->getRepository(); | ||||||
|  |     $viewer = $this->getViewer(); | ||||||
|  |     $action_list = $this->newActionList(); | ||||||
|  |  | ||||||
|  |     $can_edit = PhabricatorPolicyFilter::hasCapability( | ||||||
|  |       $viewer, | ||||||
|  |       $repository, | ||||||
|  |       PhabricatorPolicyCapability::CAN_EDIT); | ||||||
|  |  | ||||||
|  |     $limits_uri = $this->getEditPageURI(); | ||||||
|  |  | ||||||
|  |     $action_list->addAction( | ||||||
|  |       id(new PhabricatorActionView()) | ||||||
|  |         ->setIcon('fa-pencil') | ||||||
|  |         ->setName(pht('Edit Limits')) | ||||||
|  |         ->setHref($limits_uri) | ||||||
|  |         ->setDisabled(!$can_edit) | ||||||
|  |         ->setWorkflow(!$can_edit)); | ||||||
|  |  | ||||||
|  |     return $this->newCurtainView() | ||||||
|  |       ->setActionList($action_list); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function buildManagementPanelContent() { | ||||||
|  |     $repository = $this->getRepository(); | ||||||
|  |     $viewer = $this->getViewer(); | ||||||
|  |  | ||||||
|  |     $view = id(new PHUIPropertyListView()) | ||||||
|  |       ->setViewer($viewer); | ||||||
|  |  | ||||||
|  |     $byte_limit = $repository->getFilesizeLimit(); | ||||||
|  |     if ($byte_limit) { | ||||||
|  |       $filesize_display = pht('%s Bytes', new PhutilNumber($byte_limit)); | ||||||
|  |     } else { | ||||||
|  |       $filesize_display = pht('Unlimited'); | ||||||
|  |       $filesize_display = phutil_tag('em', array(), $filesize_display); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     $view->addProperty(pht('Filesize Limit'), $filesize_display); | ||||||
|  |  | ||||||
|  |     $copy_limit = $repository->getCopyTimeLimit(); | ||||||
|  |     if ($copy_limit) { | ||||||
|  |       $copy_display = pht('%s Seconds', new PhutilNumber($copy_limit)); | ||||||
|  |     } else { | ||||||
|  |       $copy_default = $repository->getDefaultCopyTimeLimit(); | ||||||
|  |       $copy_display = pht( | ||||||
|  |         'Default (%s Seconds)', | ||||||
|  |         new PhutilNumber($copy_default)); | ||||||
|  |       $copy_display = phutil_tag('em', array(), $copy_display); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     $view->addProperty(pht('Clone/Fetch Timeout'), $copy_display); | ||||||
|  |  | ||||||
|  |     return $this->newBox(pht('Limits'), $view); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -139,7 +139,7 @@ abstract class DiffusionCommandEngine extends Phobject { | |||||||
|     // to try to avoid cases where `git fetch` hangs for some reason and we're |     // to try to avoid cases where `git fetch` hangs for some reason and we're | ||||||
|     // left sitting with a held lock forever. |     // left sitting with a held lock forever. | ||||||
|     $repository = $this->getRepository(); |     $repository = $this->getRepository(); | ||||||
|     $future->setTimeout($repository->getCopyTimeLimit()); |     $future->setTimeout($repository->getEffectiveCopyTimeLimit()); | ||||||
|  |  | ||||||
|     return $future; |     return $future; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -187,7 +187,7 @@ final class DrydockWorkingCopyBlueprintImplementation | |||||||
|         (string)$repository->getCloneURIObject(), |         (string)$repository->getCloneURIObject(), | ||||||
|         $path); |         $path); | ||||||
|  |  | ||||||
|       $future->setTimeout($repository->getCopyTimeLimit()); |       $future->setTimeout($repository->getEffectiveCopyTimeLimit()); | ||||||
|  |  | ||||||
|       $futures[$directory] = $future; |       $futures[$directory] = $future; | ||||||
|     } |     } | ||||||
| @@ -284,7 +284,7 @@ final class DrydockWorkingCopyBlueprintImplementation | |||||||
|       } |       } | ||||||
|  |  | ||||||
|       $this->newExecvFuture($interface, $cmd, $arg) |       $this->newExecvFuture($interface, $cmd, $arg) | ||||||
|         ->setTimeout($repository->getCopyTimeLimit()) |         ->setTimeout($repository->getEffectiveCopyTimeLimit()) | ||||||
|         ->resolvex(); |         ->resolvex(); | ||||||
|  |  | ||||||
|       if (idx($spec, 'default')) { |       if (idx($spec, 'default')) { | ||||||
| @@ -310,7 +310,7 @@ final class DrydockWorkingCopyBlueprintImplementation | |||||||
|  |  | ||||||
|         try { |         try { | ||||||
|           $this->newExecvFuture($interface, $cmd, $arg) |           $this->newExecvFuture($interface, $cmd, $arg) | ||||||
|             ->setTimeout($repository->getCopyTimeLimit()) |             ->setTimeout($repository->getEffectiveCopyTimeLimit()) | ||||||
|             ->resolvex(); |             ->resolvex(); | ||||||
|         } catch (CommandException $ex) { |         } catch (CommandException $ex) { | ||||||
|           $display_command = csprintf( |           $display_command = csprintf( | ||||||
|   | |||||||
| @@ -1898,9 +1898,33 @@ final class PhabricatorRepository extends PhabricatorRepositoryDAO | |||||||
|    * @return int Maximum number of seconds to spend copying this repository. |    * @return int Maximum number of seconds to spend copying this repository. | ||||||
|    */ |    */ | ||||||
|   public function getCopyTimeLimit() { |   public function getCopyTimeLimit() { | ||||||
|  |     return $this->getDetail('limit.copy'); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function setCopyTimeLimit($limit) { | ||||||
|  |     return $this->setDetail('limit.copy', $limit); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function getDefaultCopyTimeLimit() { | ||||||
|     return phutil_units('15 minutes in seconds'); |     return phutil_units('15 minutes in seconds'); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   public function getEffectiveCopyTimeLimit() { | ||||||
|  |     $limit = $this->getCopyTimeLimit(); | ||||||
|  |     if ($limit) { | ||||||
|  |       return $limit; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return $this->getDefaultCopyTimeLimit(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function getFilesizeLimit() { | ||||||
|  |     return $this->getDetail('limit.filesize'); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function setFilesizeLimit($limit) { | ||||||
|  |     return $this->setDetail('limit.filesize', $limit); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
|    * Retrieve the service URI for the device hosting this repository. |    * Retrieve the service URI for the device hosting this repository. | ||||||
|   | |||||||
| @@ -0,0 +1,77 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | final class PhabricatorRepositoryCopyTimeLimitTransaction | ||||||
|  |   extends PhabricatorRepositoryTransactionType { | ||||||
|  |  | ||||||
|  |   const TRANSACTIONTYPE = 'limit.copy'; | ||||||
|  |  | ||||||
|  |   public function generateOldValue($object) { | ||||||
|  |     return $object->getCopyTimeLimit(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function generateNewValue($object, $value) { | ||||||
|  |     if (!strlen($value)) { | ||||||
|  |       return null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     $value = (int)$value; | ||||||
|  |     if (!$value) { | ||||||
|  |       return null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return $value; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function applyInternalEffects($object, $value) { | ||||||
|  |     $object->setCopyTimeLimit($value); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function getTitle() { | ||||||
|  |     $old = $this->getOldValue(); | ||||||
|  |     $new = $this->getNewValue(); | ||||||
|  |  | ||||||
|  |     if ($old && $new) { | ||||||
|  |       return pht( | ||||||
|  |         '%s changed the copy time limit for this repository from %s seconds '. | ||||||
|  |         'to %s seconds.', | ||||||
|  |         $this->renderAuthor(), | ||||||
|  |         $this->renderOldValue(), | ||||||
|  |         $this->renderNewValue()); | ||||||
|  |     } else if ($new) { | ||||||
|  |       return pht( | ||||||
|  |         '%s set the copy time limit for this repository to %s seconds.', | ||||||
|  |         $this->renderAuthor(), | ||||||
|  |         $this->renderNewValue()); | ||||||
|  |     } else { | ||||||
|  |       return pht( | ||||||
|  |         '%s reset the copy time limit (%s seconds) for this repository '. | ||||||
|  |         'to the default value.', | ||||||
|  |         $this->renderAuthor(), | ||||||
|  |         $this->renderOldValue()); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function validateTransactions($object, array $xactions) { | ||||||
|  |     $errors = array(); | ||||||
|  |  | ||||||
|  |     foreach ($xactions as $xaction) { | ||||||
|  |       $new = $xaction->getNewValue(); | ||||||
|  |  | ||||||
|  |       if (!strlen($new)) { | ||||||
|  |         continue; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (!preg_match('/^\d+\z/', $new)) { | ||||||
|  |         $errors[] = $this->newInvalidError( | ||||||
|  |           pht( | ||||||
|  |             'Unable to parse copy time limit, specify a positive number '. | ||||||
|  |             'of seconds.'), | ||||||
|  |           $xaction); | ||||||
|  |         continue; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return $errors; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,78 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | final class PhabricatorRepositoryFilesizeLimitTransaction | ||||||
|  |   extends PhabricatorRepositoryTransactionType { | ||||||
|  |  | ||||||
|  |   const TRANSACTIONTYPE = 'limit.filesize'; | ||||||
|  |  | ||||||
|  |   public function generateOldValue($object) { | ||||||
|  |     return $object->getFilesizeLimit(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function generateNewValue($object, $value) { | ||||||
|  |     if (!strlen($value)) { | ||||||
|  |       return null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     $value = phutil_parse_bytes($value); | ||||||
|  |     if (!$value) { | ||||||
|  |       return null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return $value; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function applyInternalEffects($object, $value) { | ||||||
|  |     $object->setFilesizeLimit($value); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function getTitle() { | ||||||
|  |     $old = $this->getOldValue(); | ||||||
|  |     $new = $this->getNewValue(); | ||||||
|  |  | ||||||
|  |     if ($old && $new) { | ||||||
|  |       return pht( | ||||||
|  |         '%s changed the filesize limit for this repository from %s bytes to '. | ||||||
|  |         '%s bytes.', | ||||||
|  |         $this->renderAuthor(), | ||||||
|  |         $this->renderOldValue(), | ||||||
|  |         $this->renderNewValue()); | ||||||
|  |     } else if ($new) { | ||||||
|  |       return pht( | ||||||
|  |         '%s set the filesize limit for this repository to %s bytes.', | ||||||
|  |         $this->renderAuthor(), | ||||||
|  |         $this->renderNewValue()); | ||||||
|  |     } else { | ||||||
|  |       return pht( | ||||||
|  |         '%s removed the filesize limit (%s bytes) for this repository.', | ||||||
|  |         $this->renderAuthor(), | ||||||
|  |         $this->renderOldValue()); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function validateTransactions($object, array $xactions) { | ||||||
|  |     $errors = array(); | ||||||
|  |  | ||||||
|  |     foreach ($xactions as $xaction) { | ||||||
|  |       $new = $xaction->getNewValue(); | ||||||
|  |  | ||||||
|  |       if (!strlen($new)) { | ||||||
|  |         continue; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       try { | ||||||
|  |         $value = phutil_parse_bytes($new); | ||||||
|  |       } catch (Exception $ex) { | ||||||
|  |         $errors[] = $this->newInvalidError( | ||||||
|  |           pht( | ||||||
|  |             'Unable to parse filesize limit: %s', | ||||||
|  |             $ex->getMessage()), | ||||||
|  |           $xaction); | ||||||
|  |         continue; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return $errors; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -236,37 +236,20 @@ fetch from, serve from, and push to. | |||||||
| These options are covered in detail in @{article:Diffusion User Guide: URIs}. | These options are covered in detail in @{article:Diffusion User Guide: URIs}. | ||||||
|  |  | ||||||
|  |  | ||||||
| Staging Area | Limits | ||||||
| ============ |  | ||||||
|  |  | ||||||
| The **Staging Area** panel configures staging areas, used to make proposed |  | ||||||
| changes available to build and continuous integration systems. |  | ||||||
|  |  | ||||||
| For more details, see @{article:Harbormaster User Guide}. |  | ||||||
|  |  | ||||||
|  |  | ||||||
| Automation |  | ||||||
| ========== |  | ||||||
|  |  | ||||||
| The **Automation** panel configures support for allowing Phabricator to make |  | ||||||
| writes directly to the repository, so that it can perform operations like |  | ||||||
| automatically landing revisions from the web UI. |  | ||||||
|  |  | ||||||
| For details on repository automation, see |  | ||||||
| @{article:Drydock User Guide: Repository Automation}. |  | ||||||
|  |  | ||||||
|  |  | ||||||
| Symbols |  | ||||||
| ====== | ====== | ||||||
|  |  | ||||||
| The **Symbols** panel allows you to customize how symbols (like class and | The **Limits** panel allows you to configure limits and timeouts. | ||||||
| function names) are linked when viewing code in the repository, and when |  | ||||||
| viewing revisions which propose code changes to the repository. |  | ||||||
|  |  | ||||||
| To take advantage of this feature, you need to do additional work to build | **Filesize Limit**: Allows you to set a maximum filesize for any file in the | ||||||
| symbol indexes. For details on configuring and populating symbol indexes, see | repository. If a commit creates a larger file (or modifies an existing file so | ||||||
| @{article:User Guide: Symbol Indexes}. | it becomes too large) it will be rejected. This option only applies to hosted | ||||||
|  | repositories. | ||||||
|  |  | ||||||
|  | **Clone/Fetch Timeout**: Configure the internal timeout for creating copies | ||||||
|  | of this repository during operations like intracluster synchronization and | ||||||
|  | Drydock working copy construction. This timeout does not affect external | ||||||
|  | users. | ||||||
|  |  | ||||||
| Branches | Branches | ||||||
| ======== | ======== | ||||||
| @@ -311,6 +294,38 @@ revisions and tasks. If you don't want Phabricator to close objects when it | |||||||
| discovers new commits, disable **Autoclose** for the repository. | discovers new commits, disable **Autoclose** for the repository. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Staging Area | ||||||
|  | ============ | ||||||
|  |  | ||||||
|  | The **Staging Area** panel configures staging areas, used to make proposed | ||||||
|  | changes available to build and continuous integration systems. | ||||||
|  |  | ||||||
|  | For more details, see @{article:Harbormaster User Guide}. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Automation | ||||||
|  | ========== | ||||||
|  |  | ||||||
|  | The **Automation** panel configures support for allowing Phabricator to make | ||||||
|  | writes directly to the repository, so that it can perform operations like | ||||||
|  | automatically landing revisions from the web UI. | ||||||
|  |  | ||||||
|  | For details on repository automation, see | ||||||
|  | @{article:Drydock User Guide: Repository Automation}. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Symbols | ||||||
|  | ====== | ||||||
|  |  | ||||||
|  | The **Symbols** panel allows you to customize how symbols (like class and | ||||||
|  | function names) are linked when viewing code in the repository, and when | ||||||
|  | viewing revisions which propose code changes to the repository. | ||||||
|  |  | ||||||
|  | To take advantage of this feature, you need to do additional work to build | ||||||
|  | symbol indexes. For details on configuring and populating symbol indexes, see | ||||||
|  | @{article:User Guide: Symbol Indexes}. | ||||||
|  |  | ||||||
|  |  | ||||||
| Repository Identifiers and Names | Repository Identifiers and Names | ||||||
| ================================ | ================================ | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 epriestley
					epriestley