2011-07-22 19:17:57 +02:00
|
|
|
<?php
|
|
|
|
|
2012-08-13 21:37:26 +02:00
|
|
|
final class PhabricatorSettingsPanelSSHKeys
|
|
|
|
extends PhabricatorSettingsPanel {
|
2011-07-22 19:17:57 +02:00
|
|
|
|
2012-08-13 21:37:26 +02:00
|
|
|
public function getPanelKey() {
|
|
|
|
return 'ssh';
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getPanelName() {
|
|
|
|
return pht('SSH Public Keys');
|
|
|
|
}
|
2011-07-22 19:17:57 +02:00
|
|
|
|
2012-08-13 21:37:26 +02:00
|
|
|
public function getPanelGroup() {
|
|
|
|
return pht('Authentication');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function isEnabled() {
|
2011-07-24 20:02:08 +02:00
|
|
|
return PhabricatorEnv::getEnvConfig('auth.sshkeys.enabled');
|
|
|
|
}
|
|
|
|
|
2012-08-13 21:37:26 +02:00
|
|
|
public function processRequest(AphrontRequest $request) {
|
2011-07-24 20:02:08 +02:00
|
|
|
|
2011-07-22 19:17:57 +02:00
|
|
|
$user = $request->getUser();
|
|
|
|
|
|
|
|
$edit = $request->getStr('edit');
|
|
|
|
$delete = $request->getStr('delete');
|
|
|
|
if (!$edit && !$delete) {
|
2012-08-13 21:37:26 +02:00
|
|
|
return $this->renderKeyListView($request);
|
2011-07-22 19:17:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$id = nonempty($edit, $delete);
|
|
|
|
|
|
|
|
if ($id && is_numeric($id)) {
|
|
|
|
// NOTE: Prevent editing/deleting of keys you don't own.
|
|
|
|
$key = id(new PhabricatorUserSSHKey())->loadOneWhere(
|
|
|
|
'userPHID = %s AND id = %d',
|
|
|
|
$user->getPHID(),
|
2012-08-13 21:37:26 +02:00
|
|
|
(int)$id);
|
2011-07-22 19:17:57 +02:00
|
|
|
if (!$key) {
|
|
|
|
return new Aphront404Response();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$key = new PhabricatorUserSSHKey();
|
|
|
|
$key->setUserPHID($user->getPHID());
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($delete) {
|
2012-08-13 21:37:26 +02:00
|
|
|
return $this->processDelete($request, $key);
|
2011-07-22 19:17:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$e_name = true;
|
|
|
|
$e_key = true;
|
|
|
|
$errors = array();
|
|
|
|
$entire_key = $key->getEntireKey();
|
|
|
|
if ($request->isFormPost()) {
|
|
|
|
$key->setName($request->getStr('name'));
|
|
|
|
$entire_key = $request->getStr('key');
|
|
|
|
|
|
|
|
if (!strlen($entire_key)) {
|
2013-03-03 15:52:42 +01:00
|
|
|
$errors[] = pht('You must provide an SSH Public Key.');
|
|
|
|
$e_key = pht('Required');
|
2011-07-22 19:17:57 +02:00
|
|
|
} else {
|
|
|
|
$parts = str_replace("\n", '', trim($entire_key));
|
|
|
|
$parts = preg_split('/\s+/', $parts);
|
|
|
|
if (count($parts) == 2) {
|
|
|
|
$parts[] = ''; // Add an empty comment part.
|
|
|
|
} else if (count($parts) == 3) {
|
|
|
|
// This is the expected case.
|
|
|
|
} else {
|
|
|
|
if (preg_match('/private\s*key/i', $entire_key)) {
|
|
|
|
// Try to give the user a better error message if it looks like
|
|
|
|
// they uploaded a private key.
|
2013-03-03 15:52:42 +01:00
|
|
|
$e_key = pht('Invalid');
|
|
|
|
$errors[] = pht('Provide your public key, not your private key!');
|
2011-07-22 19:17:57 +02:00
|
|
|
} else {
|
2013-03-03 15:52:42 +01:00
|
|
|
$e_key = pht('Invalid');
|
|
|
|
$errors[] = pht('Provided public key is not properly formatted.');
|
2011-07-22 19:17:57 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$errors) {
|
|
|
|
list($type, $body, $comment) = $parts;
|
|
|
|
if (!preg_match('/^ssh-dsa|ssh-rsa$/', $type)) {
|
2013-03-03 15:52:42 +01:00
|
|
|
$e_key = pht('Invalid');
|
|
|
|
$errors[] = pht('Public key should be "ssh-dsa" or "ssh-rsa".');
|
2011-07-22 19:17:57 +02:00
|
|
|
} else {
|
|
|
|
$key->setKeyType($type);
|
|
|
|
$key->setKeyBody($body);
|
2012-03-10 03:06:39 +01:00
|
|
|
$key->setKeyHash(md5($body));
|
2011-07-22 19:17:57 +02:00
|
|
|
$key->setKeyComment($comment);
|
|
|
|
|
|
|
|
$e_key = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strlen($key->getName())) {
|
2013-03-03 15:52:42 +01:00
|
|
|
$errors[] = pht('You must name this public key.');
|
|
|
|
$e_name = pht('Required');
|
2011-07-22 19:17:57 +02:00
|
|
|
} else {
|
|
|
|
$e_name = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$errors) {
|
|
|
|
try {
|
|
|
|
$key->save();
|
|
|
|
return id(new AphrontRedirectResponse())
|
2012-08-13 21:37:26 +02:00
|
|
|
->setURI($this->getPanelURI());
|
2011-07-22 19:17:57 +02:00
|
|
|
} catch (AphrontQueryDuplicateKeyException $ex) {
|
2013-03-03 15:52:42 +01:00
|
|
|
$e_key = pht('Duplicate');
|
|
|
|
$errors[] = pht('This public key is already associated with a user '.
|
|
|
|
'account.');
|
2011-07-22 19:17:57 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$error_view = null;
|
|
|
|
if ($errors) {
|
|
|
|
$error_view = new AphrontErrorView();
|
2013-03-03 15:52:42 +01:00
|
|
|
$error_view->setTitle(pht('Form Errors'));
|
2011-07-22 19:17:57 +02:00
|
|
|
$error_view->setErrors($errors);
|
|
|
|
}
|
|
|
|
|
|
|
|
$is_new = !$key->getID();
|
|
|
|
|
|
|
|
if ($is_new) {
|
2013-03-03 15:52:42 +01:00
|
|
|
$header = pht('Add New SSH Public Key');
|
|
|
|
$save = pht('Add Key');
|
2011-07-22 19:17:57 +02:00
|
|
|
} else {
|
2013-03-03 15:52:42 +01:00
|
|
|
$header = pht('Edit SSH Public Key');
|
|
|
|
$save = pht('Save Changes');
|
2011-07-22 19:17:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$form = id(new AphrontFormView())
|
|
|
|
->setUser($user)
|
|
|
|
->addHiddenInput('edit', $is_new ? 'true' : $key->getID())
|
|
|
|
->appendChild(
|
|
|
|
id(new AphrontFormTextControl())
|
2013-03-03 15:52:42 +01:00
|
|
|
->setLabel(pht('Name'))
|
2011-07-22 19:17:57 +02:00
|
|
|
->setName('name')
|
|
|
|
->setValue($key->getName())
|
|
|
|
->setError($e_name))
|
|
|
|
->appendChild(
|
|
|
|
id(new AphrontFormTextAreaControl())
|
2013-03-03 15:52:42 +01:00
|
|
|
->setLabel(pht('Public Key'))
|
2011-07-22 19:17:57 +02:00
|
|
|
->setName('key')
|
|
|
|
->setValue($entire_key)
|
|
|
|
->setError($e_key))
|
|
|
|
->appendChild(
|
|
|
|
id(new AphrontFormSubmitControl())
|
2012-08-13 21:37:26 +02:00
|
|
|
->addCancelButton($this->getPanelURI())
|
2011-07-22 19:17:57 +02:00
|
|
|
->setValue($save));
|
|
|
|
|
2013-05-08 03:08:12 +02:00
|
|
|
$header_title = new PhabricatorHeaderView();
|
|
|
|
$header_title->setHeader($header);
|
2011-07-22 19:17:57 +02:00
|
|
|
|
|
|
|
return id(new AphrontNullView())
|
|
|
|
->appendChild(
|
|
|
|
array(
|
|
|
|
$error_view,
|
2013-05-08 03:08:12 +02:00
|
|
|
$header_title,
|
|
|
|
$form,
|
2011-07-22 19:17:57 +02:00
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2012-08-13 21:37:26 +02:00
|
|
|
private function renderKeyListView(AphrontRequest $request) {
|
|
|
|
|
2011-07-22 19:17:57 +02:00
|
|
|
$user = $request->getUser();
|
|
|
|
|
|
|
|
$keys = id(new PhabricatorUserSSHKey())->loadAllWhere(
|
|
|
|
'userPHID = %s',
|
|
|
|
$user->getPHID());
|
|
|
|
|
|
|
|
$rows = array();
|
|
|
|
foreach ($keys as $key) {
|
|
|
|
$rows[] = array(
|
2013-01-18 03:43:35 +01:00
|
|
|
phutil_tag(
|
2011-07-22 19:17:57 +02:00
|
|
|
'a',
|
|
|
|
array(
|
2012-08-13 21:37:26 +02:00
|
|
|
'href' => $this->getPanelURI('?edit='.$key->getID()),
|
2011-07-22 19:17:57 +02:00
|
|
|
),
|
2013-01-18 03:43:35 +01:00
|
|
|
$key->getName()),
|
2013-02-13 23:50:15 +01:00
|
|
|
$key->getKeyComment(),
|
|
|
|
$key->getKeyType(),
|
2011-07-22 19:17:57 +02:00
|
|
|
phabricator_date($key->getDateCreated(), $user),
|
|
|
|
phabricator_time($key->getDateCreated(), $user),
|
2013-01-25 21:57:17 +01:00
|
|
|
javelin_tag(
|
2011-07-22 19:17:57 +02:00
|
|
|
'a',
|
|
|
|
array(
|
2012-08-13 21:37:26 +02:00
|
|
|
'href' => $this->getPanelURI('?delete='.$key->getID()),
|
2011-07-22 19:17:57 +02:00
|
|
|
'class' => 'small grey button',
|
|
|
|
'sigil' => 'workflow',
|
|
|
|
),
|
2013-03-03 15:52:42 +01:00
|
|
|
pht('Delete')),
|
2011-07-22 19:17:57 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$table = new AphrontTableView($rows);
|
2013-03-03 15:52:42 +01:00
|
|
|
$table->setNoDataString(pht("You haven't added any SSH Public Keys."));
|
2011-07-22 19:17:57 +02:00
|
|
|
$table->setHeaders(
|
|
|
|
array(
|
2013-03-03 15:52:42 +01:00
|
|
|
pht('Name'),
|
|
|
|
pht('Comment'),
|
|
|
|
pht('Type'),
|
|
|
|
pht('Created'),
|
|
|
|
pht('Time'),
|
2011-07-22 19:17:57 +02:00
|
|
|
'',
|
|
|
|
));
|
|
|
|
$table->setColumnClasses(
|
|
|
|
array(
|
|
|
|
'wide pri',
|
|
|
|
'',
|
|
|
|
'',
|
|
|
|
'',
|
|
|
|
'right',
|
|
|
|
'action',
|
|
|
|
));
|
|
|
|
|
|
|
|
$panel = new AphrontPanelView();
|
|
|
|
$panel->addButton(
|
2013-01-18 03:57:09 +01:00
|
|
|
phutil_tag(
|
2011-07-22 19:17:57 +02:00
|
|
|
'a',
|
|
|
|
array(
|
2012-08-13 21:37:26 +02:00
|
|
|
'href' => $this->getPanelURI('?edit=true'),
|
2011-07-22 19:17:57 +02:00
|
|
|
'class' => 'green button',
|
|
|
|
),
|
2013-03-03 15:52:42 +01:00
|
|
|
pht('Add New Public Key')));
|
|
|
|
$panel->setHeader(pht('SSH Public Keys'));
|
2011-07-22 19:17:57 +02:00
|
|
|
$panel->appendChild($table);
|
2013-01-16 16:49:05 +01:00
|
|
|
$panel->setNoBackground();
|
2011-07-22 19:17:57 +02:00
|
|
|
|
|
|
|
return $panel;
|
|
|
|
}
|
|
|
|
|
2012-08-13 21:37:26 +02:00
|
|
|
private function processDelete(
|
|
|
|
AphrontRequest $request,
|
|
|
|
PhabricatorUserSSHKey $key) {
|
|
|
|
|
2011-07-22 19:17:57 +02:00
|
|
|
$user = $request->getUser();
|
|
|
|
|
2013-02-13 23:50:15 +01:00
|
|
|
$name = phutil_tag('strong', array(), $key->getName());
|
2011-07-22 19:17:57 +02:00
|
|
|
|
|
|
|
if ($request->isDialogFormPost()) {
|
|
|
|
$key->delete();
|
|
|
|
return id(new AphrontReloadResponse())
|
2012-08-13 21:37:26 +02:00
|
|
|
->setURI($this->getPanelURI());
|
2011-07-22 19:17:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$dialog = id(new AphrontDialogView())
|
|
|
|
->setUser($user)
|
|
|
|
->addHiddenInput('delete', $key->getID())
|
2013-03-03 15:52:42 +01:00
|
|
|
->setTitle(pht('Really delete SSH Public Key?'))
|
2013-02-13 23:50:15 +01:00
|
|
|
->appendChild(phutil_tag('p', array(), pht(
|
|
|
|
'The key "%s" will be permanently deleted, and you will not longer be '.
|
|
|
|
'able to use the corresponding private key to authenticate.',
|
|
|
|
$name)))
|
2013-03-03 15:52:42 +01:00
|
|
|
->addSubmitButton(pht('Delete Public Key'))
|
2012-08-13 21:37:26 +02:00
|
|
|
->addCancelButton($this->getPanelURI());
|
2011-07-22 19:17:57 +02:00
|
|
|
|
|
|
|
return id(new AphrontDialogResponse())
|
|
|
|
->setDialog($dialog);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|