Skip to content
Snippets Groups Projects
Commit ced7aad2 authored by Peter Rotich's avatar Peter Rotich
Browse files

Merge pull request #1808 from greezybacon/feature/user-staff-cli


cli: user: Add list, activate, lock features

Reviewed-By: default avatarPeter Rotich <peter@osticket.com>
parents a468f8a8 d7b61fd4
No related branches found
No related tags found
No related merge requests found
...@@ -462,14 +462,22 @@ class User extends UserModel { ...@@ -462,14 +462,22 @@ class User extends UserModel {
$users[] = $data; $users[] = $data;
} }
db_autocommit(false);
$error = false;
foreach ($users as $u) { foreach ($users as $u) {
$vars = array_combine($keys, $u); $vars = array_combine($keys, $u);
if (!static::fromVars($vars)) if (!static::fromVars($vars)) {
return sprintf(__('Unable to import user: %s'), $error = sprintf(__('Unable to import user: %s'),
print_r($vars, true)); print_r($vars, true));
break;
}
} }
if ($error)
db_rollback();
db_autocommit(true);
return count($users); return $error ?: count($users);
} }
function importFromPost($stuff, $extra=array()) { function importFromPost($stuff, $extra=array()) {
...@@ -937,6 +945,10 @@ class UserAccount extends UserAccountModel { ...@@ -937,6 +945,10 @@ class UserAccount extends UserAccountModel {
return static::sendUnlockEmail('registration-client') === true; return static::sendUnlockEmail('registration-client') === true;
} }
function setPassword($new) {
$this->set('passwd', Passwd::hash($new));
}
protected function sendUnlockEmail($template) { protected function sendUnlockEmail($template) {
global $ost, $cfg; global $ost, $cfg;
...@@ -1021,7 +1033,7 @@ class UserAccount extends UserAccountModel { ...@@ -1021,7 +1033,7 @@ class UserAccount extends UserAccountModel {
$this->set('username', $vars['username']); $this->set('username', $vars['username']);
if ($vars['passwd1']) { if ($vars['passwd1']) {
$this->set('passwd', Passwd::hash($vars['passwd1'])); $this->setPassword($vars['passwd1']);
$this->setStatus(UserAccountStatus::CONFIRMED); $this->setStatus(UserAccountStatus::CONFIRMED);
} }
......
...@@ -90,6 +90,12 @@ function db_autocommit($enable=true) { ...@@ -90,6 +90,12 @@ function db_autocommit($enable=true) {
return $__db->autocommit($enable); return $__db->autocommit($enable);
} }
function db_rollback() {
global $__db;
return $__db->rollback();
}
function db_close() { function db_close() {
global $__db; global $__db;
return @$__db->close(); return @$__db->close();
......
...@@ -39,8 +39,14 @@ class Option { ...@@ -39,8 +39,14 @@ class Option {
$value = null; $value = null;
elseif ($value) elseif ($value)
$nargs = 1; $nargs = 1;
if ($this->type == 'int') switch ($this->type) {
case 'int':
$value = (int)$value; $value = (int)$value;
break;
case 'bool':
$value = strcasecmp($value, 'true') === 0 || ((int) $value);
break;
}
switch ($this->action) { switch ($this->action) {
case 'store_true': case 'store_true':
$value = true; $value = true;
...@@ -156,7 +162,7 @@ class Module { ...@@ -156,7 +162,7 @@ class Module {
if ($this->arguments) { if ($this->arguments) {
echo "\nArguments:\n"; echo "\nArguments:\n";
foreach ($this->arguments as $name=>$help) foreach ($this->arguments as $name=>$help) {
$extra = ''; $extra = '';
if (is_array($help)) { if (is_array($help)) {
if (isset($help['options']) && is_array($help['options'])) { if (isset($help['options']) && is_array($help['options'])) {
...@@ -169,6 +175,7 @@ class Module { ...@@ -169,6 +175,7 @@ class Module {
echo $name . "\n " . wordwrap( echo $name . "\n " . wordwrap(
preg_replace('/\s+/', ' ', $help), 76, "\n ") preg_replace('/\s+/', ' ', $help), 76, "\n ")
.$extra."\n"; .$extra."\n";
}
} }
if ($this->epilog) { if ($this->epilog) {
......
<?php
require_once dirname(__file__) . "/class.module.php";
require_once dirname(__file__) . "/../cli.inc.php";
class CronManager extends Module {
var $prologue = 'CLI cron manager for osTicket';
var $arguments = array(
'action' => array(
'help' => 'Action to be performed',
'options' => array(
'fetch' => 'Fetch email',
'search' => 'Build search index'
),
),
);
function run($args, $options) {
Bootstrap::connect();
$ost = osTicket::start();
switch (strtolower($args[0])) {
case 'fetch':
Cron::MailFetcher();
break;
case 'search':
$ost->searcher->backend->IndexOldStuff();
break;
}
}
}
Module::register('cron', 'CronManager');
?>
...@@ -9,8 +9,12 @@ class UserManager extends Module { ...@@ -9,8 +9,12 @@ class UserManager extends Module {
'action' => array( 'action' => array(
'help' => 'Action to be performed', 'help' => 'Action to be performed',
'options' => array( 'options' => array(
'import' => 'Import users to the system', 'import' => 'Import users from CSV file',
'export' => 'Export users from the system', 'export' => 'Export users from the system to CSV',
'activate' => 'Create or activate an account',
'lock' => "Lock a user's account",
'set-password' => "Set a user's account password",
'list' => 'List users based on search criteria',
), ),
), ),
); );
...@@ -21,6 +25,26 @@ class UserManager extends Module { ...@@ -21,6 +25,26 @@ class UserManager extends Module {
'help' => 'File or stream to process'), 'help' => 'File or stream to process'),
'org' => array('-O', '--org', 'metavar'=>'ORGID', 'org' => array('-O', '--org', 'metavar'=>'ORGID',
'help' => 'Set the organization ID on import'), 'help' => 'Set the organization ID on import'),
'send-mail' => array('-m', '--send-mail',
'help' => 'Send the user an email. Used with `activate` and `set-password`',
'default' => false, 'action' => 'store_true'),
'verbose' => array('-v', '--verbose', 'default'=>false,
'action'=>'store_true', 'help' => 'Be more verbose'
),
// -- Search criteria
'account' => array('-A', '--account', 'type'=>'bool', 'metavar'=>'bool',
'help' => 'Search for users based on activation status'),
'isconfirmed' => array('-C', '--isconfirmed', 'type'=>'bool', 'metavar'=>'bool',
'help' => 'Search for users based on confirmation status'),
'islocked' => array('-L', '--islocked', 'type'=>'bool', 'metavar'=>'bool',
'help' => 'Search for users based on locked status'),
'email' => array('-E', '--email',
'help' => 'Search by email address'),
'id' => array('-U', '--id',
'help' => 'Search by user id'),
); );
var $stream; var $stream;
...@@ -30,42 +54,164 @@ class UserManager extends Module { ...@@ -30,42 +54,164 @@ class UserManager extends Module {
Bootstrap::connect(); Bootstrap::connect();
switch ($args['action']) { switch ($args['action']) {
case 'import': case 'import':
// Properly detect Macintosh style line endings // Properly detect Macintosh style line endings
ini_set('auto_detect_line_endings', true); ini_set('auto_detect_line_endings', true);
if (!$options['file']) if (!$options['file'])
$this->fail('CSV file to import users from is required!'); $this->fail('CSV file to import users from is required!');
elseif (!($this->stream = fopen($options['file'], 'rb'))) elseif (!($this->stream = fopen($options['file'], 'rb')))
$this->fail("Unable to open input file [{$options['file']}]"); $this->fail("Unable to open input file [{$options['file']}]");
$extras = array(); $extras = array();
if ($options['org']) { if ($options['org']) {
if (!($org = Organization::lookup($options['org']))) if (!($org = Organization::lookup($options['org'])))
$this->fail($options['org'].': Unknown organization ID'); $this->fail($options['org'].': Unknown organization ID');
$extras['org_id'] = $options['org']; $extras['org_id'] = $options['org'];
}
$status = User::importCsv($this->stream, $extras);
if (is_numeric($status))
$this->stderr->write("Successfully imported $status clients\n");
else
$this->fail($status);
break;
case 'export':
$stream = $options['file'] ?: 'php://stdout';
if (!($this->stream = fopen($stream, 'c')))
$this->fail("Unable to open output file [{$options['file']}]");
fputcsv($this->stream, array('Name', 'Email'));
foreach (User::objects() as $user)
fputcsv($this->stream,
array((string) $user->getName(), $user->getEmail()));
break;
case 'activate':
$users = $this->getQuerySet($options);
foreach ($users as $U) {
if ($options['verbose']) {
$this->stderr->write(sprintf(
"Activating %s <%s>\n",
$U->getName(), $U->getDefaultEmail()
));
} }
$status = User::importCsv($this->stream, $extras); if (!($account = $U->getAccount())) {
if (is_numeric($status)) $account = UserAccount::create(array('user' => $U));
$this->stderr->write("Successfully imported $status clients\n"); $U->account = $account;
else $U->save();
$this->fail($status); }
if ($options['send-mail']) {
global $ost, $cfg;
$ost = osTicket::start();
$cfg = $ost->getConfig();
if (($error = $account->sendConfirmEmail()) && $error !== true) {
$this->warn(sprintf('%s: Unable to send email: %s',
$U->getDefaultEmail(), $error->getMessage()
));
}
}
}
break;
case 'lock':
$users = $this->getQuerySet($options);
$users->select_related('account');
foreach ($users as $U) {
if (!($account = $U->getAccount())) {
$this->warn(sprintf(
'%s: User does not have a client account',
$U->getName()
));
}
$account->setFlag(UserAccountStatus::LOCKED);
$account->save();
}
break;
case 'list':
$users = $this->getQuerySet($options);
foreach ($users as $U) {
$this->stdout->write(sprintf(
"%d %s <%s>%s\n",
$U->id, $U->getName(), $U->getDefaultEmail(),
($O = $U->getOrganization()) ? " ({$O->getName()})" : ''
));
}
break;
case 'set-password':
$this->stderr->write('Enter new password: ');
$ps1 = fgets(STDIN);
if (!function_exists('posix_isatty') || !posix_isatty(STDIN)) {
$this->stderr->write('Re-enter new password: ');
$ps2 = fgets(STDIN);
if ($ps1 != $ps2)
$this->fail('Passwords do not match');
}
// Account is required
$options['account'] = true;
$users = $this->getQuerySet($options);
$updated = 0;
foreach ($users as $U) {
$U->account->setPassword($ps1);
if ($U->account->save())
$updated++;
}
$this->stdout->write(sprintf('Updated %d users', $updated));
break;
default:
$this->stderr->write('Unknown action!');
}
@fclose($this->stream);
}
function getQuerySet($options, $requireOne=false) {
$users = User::objects();
foreach ($options as $O=>$V) {
if (!isset($V))
continue;
switch ($O) {
case 'account':
$users->filter(array('account__isnull' => !$V));
break; break;
case 'export': case 'isconfirmed':
$stream = $options['file'] ?: 'php://stdout'; case 'islocked':
if (!($this->stream = fopen($stream, 'c'))) $flags = array(
$this->fail("Unable to open output file [{$options['file']}]"); 'isconfirmed' => UserAccountStatus::CONFIRMED,
'islocked' => UserAccountStatus::LOCKED,
);
$Q = new Q(array('account__status__hasbit'=>$flags[$O]));
if (!$V)
$Q->negate();
$users->filter($Q);
break;
fputcsv($this->stream, array('Name', 'Email')); case 'org':
foreach (User::objects() as $user) if (is_numeric($V))
fputcsv($this->stream, $users->filter(array('org__id'=>$V));
array((string) $user->getName(), $user->getEmail())); else
$users->filter(array('org__name__contains'=>$V));
break; break;
default:
$this->stderr->write('Unknown action!'); case 'id':
$users->filter(array('id'=>$V));
break;
}
} }
@fclose($this->stream); return $users;
} }
} }
Module::register('user', 'UserManager'); Module::register('user', 'UserManager');
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment