Made YammerImport more robust against errors; can now pause/resume/reset the import state from the admin interface.
This commit is contained in:
parent
6cf7b24017
commit
588758ed6d
@ -73,6 +73,7 @@ class YammeradminpanelAction extends AdminPanelAction
|
|||||||
{
|
{
|
||||||
// @fixme move this to saveSettings and friends?
|
// @fixme move this to saveSettings and friends?
|
||||||
if ($_SERVER['REQUEST_METHOD'] == 'POST') {
|
if ($_SERVER['REQUEST_METHOD'] == 'POST') {
|
||||||
|
StatusNet::setApi(true); // short error pages :P
|
||||||
$this->checkSessionToken();
|
$this->checkSessionToken();
|
||||||
if ($this->subaction == 'change-apikey') {
|
if ($this->subaction == 'change-apikey') {
|
||||||
$form = new YammerApiKeyForm($this);
|
$form = new YammerApiKeyForm($this);
|
||||||
@ -97,6 +98,18 @@ class YammeradminpanelAction extends AdminPanelAction
|
|||||||
$this->runner->startBackgroundImport();
|
$this->runner->startBackgroundImport();
|
||||||
|
|
||||||
$form = new YammerProgressForm($this, $this->runner);
|
$form = new YammerProgressForm($this, $this->runner);
|
||||||
|
} else if ($this->subaction == 'pause-import') {
|
||||||
|
$this->runner->recordError(_m('Paused from admin panel.'));
|
||||||
|
$form = $this->statusForm();
|
||||||
|
} else if ($this->subaction == 'continue-import') {
|
||||||
|
$this->runner->clearError();
|
||||||
|
$this->runner->startBackgroundImport();
|
||||||
|
$form = $this->statusForm();
|
||||||
|
} else if ($this->subaction == 'abort-import') {
|
||||||
|
$this->runner->reset();
|
||||||
|
$form = $this->statusForm();
|
||||||
|
} else if ($this->subaction == 'progress') {
|
||||||
|
$form = $this->statusForm();
|
||||||
} else {
|
} else {
|
||||||
throw new ClientException('Invalid POST');
|
throw new ClientException('Invalid POST');
|
||||||
}
|
}
|
||||||
|
@ -36,6 +36,7 @@ class Yammer_state extends Memcached_DataObject
|
|||||||
public $__table = 'yammer_state'; // table name
|
public $__table = 'yammer_state'; // table name
|
||||||
public $id; // int primary_key not_null
|
public $id; // int primary_key not_null
|
||||||
public $state; // import state key
|
public $state; // import state key
|
||||||
|
public $last_error; // text of last-encountered error, if any
|
||||||
public $oauth_token; // actual oauth token! clear when import is done?
|
public $oauth_token; // actual oauth token! clear when import is done?
|
||||||
public $oauth_secret; // actual oauth secret! clear when import is done?
|
public $oauth_secret; // actual oauth secret! clear when import is done?
|
||||||
public $users_page; // last page of users we've fetched
|
public $users_page; // last page of users we've fetched
|
||||||
@ -70,6 +71,7 @@ class Yammer_state extends Memcached_DataObject
|
|||||||
return array(new ColumnDef('id', 'int', null,
|
return array(new ColumnDef('id', 'int', null,
|
||||||
false, 'PRI'),
|
false, 'PRI'),
|
||||||
new ColumnDef('state', 'text'),
|
new ColumnDef('state', 'text'),
|
||||||
|
new ColumnDef('last_error', 'text'),
|
||||||
new ColumnDef('oauth_token', 'text'),
|
new ColumnDef('oauth_token', 'text'),
|
||||||
new ColumnDef('oauth_secret', 'text'),
|
new ColumnDef('oauth_secret', 'text'),
|
||||||
new ColumnDef('users_page', 'int'),
|
new ColumnDef('users_page', 'int'),
|
||||||
@ -93,6 +95,7 @@ class Yammer_state extends Memcached_DataObject
|
|||||||
{
|
{
|
||||||
return array('id' => DB_DATAOBJECT_INT + DB_DATAOBJECT_NOTNULL,
|
return array('id' => DB_DATAOBJECT_INT + DB_DATAOBJECT_NOTNULL,
|
||||||
'state' => DB_DATAOBJECT_STR,
|
'state' => DB_DATAOBJECT_STR,
|
||||||
|
'last_error' => DB_DATAOBJECT_STR,
|
||||||
'oauth_token' => DB_DATAOBJECT_STR,
|
'oauth_token' => DB_DATAOBJECT_STR,
|
||||||
'oauth_secret' => DB_DATAOBJECT_STR,
|
'oauth_secret' => DB_DATAOBJECT_STR,
|
||||||
'users_page' => DB_DATAOBJECT_INT,
|
'users_page' => DB_DATAOBJECT_INT,
|
||||||
|
@ -53,3 +53,9 @@
|
|||||||
.magiclink {
|
.magiclink {
|
||||||
margin-left: 40px;
|
margin-left: 40px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fieldset.import-error {
|
||||||
|
margin-top: 12px;
|
||||||
|
margin-bottom: 0px !important;
|
||||||
|
background-color: #fee !important;
|
||||||
|
}
|
||||||
|
@ -132,6 +132,7 @@ class YammerImporter
|
|||||||
if ($noticeId) {
|
if ($noticeId) {
|
||||||
return Notice::staticGet('id', $noticeId);
|
return Notice::staticGet('id', $noticeId);
|
||||||
} else {
|
} else {
|
||||||
|
$notice = Notice::staticGet('uri', $data['options']['uri']);
|
||||||
$content = $data['content'];
|
$content = $data['content'];
|
||||||
$user = User::staticGet($data['profile']);
|
$user = User::staticGet($data['profile']);
|
||||||
|
|
||||||
|
@ -9,7 +9,7 @@ class YammerProgressForm extends Form
|
|||||||
*/
|
*/
|
||||||
function id()
|
function id()
|
||||||
{
|
{
|
||||||
return 'yammer-progress';
|
return 'yammer-progress-form';
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -39,8 +39,11 @@ class YammerProgressForm extends Form
|
|||||||
*/
|
*/
|
||||||
function formData()
|
function formData()
|
||||||
{
|
{
|
||||||
|
$this->out->hidden('subaction', 'progress');
|
||||||
|
|
||||||
$runner = YammerRunner::init();
|
$runner = YammerRunner::init();
|
||||||
|
|
||||||
|
$error = $runner->lastError();
|
||||||
$userCount = $runner->countUsers();
|
$userCount = $runner->countUsers();
|
||||||
$groupCount = $runner->countGroups();
|
$groupCount = $runner->countGroups();
|
||||||
$fetchedCount = $runner->countFetchedNotices();
|
$fetchedCount = $runner->countFetchedNotices();
|
||||||
@ -86,7 +89,13 @@ class YammerProgressForm extends Form
|
|||||||
$steps = array_keys($labels);
|
$steps = array_keys($labels);
|
||||||
$currentStep = array_search($runner->state(), $steps);
|
$currentStep = array_search($runner->state(), $steps);
|
||||||
|
|
||||||
$this->out->elementStart('fieldset', array('class' => 'yammer-import'));
|
$classes = array('yammer-import');
|
||||||
|
if ($error) {
|
||||||
|
$classes[] = 'yammer-error';
|
||||||
|
} else {
|
||||||
|
$classes[] = 'yammer-running';
|
||||||
|
}
|
||||||
|
$this->out->elementStart('fieldset', array('class' => implode(' ', $classes)));
|
||||||
$this->out->element('legend', array(), _m('Import status'));
|
$this->out->element('legend', array(), _m('Import status'));
|
||||||
foreach ($steps as $step => $state) {
|
foreach ($steps as $step => $state) {
|
||||||
if ($state == 'init') {
|
if ($state == 'init') {
|
||||||
@ -104,7 +113,8 @@ class YammerProgressForm extends Form
|
|||||||
$this->progressBar($state,
|
$this->progressBar($state,
|
||||||
'progress',
|
'progress',
|
||||||
$labels[$state]['label'],
|
$labels[$state]['label'],
|
||||||
$labels[$state]['progress']);
|
$labels[$state]['progress'],
|
||||||
|
$error);
|
||||||
} else {
|
} else {
|
||||||
// This step has not yet been done.
|
// This step has not yet been done.
|
||||||
$this->progressBar($state,
|
$this->progressBar($state,
|
||||||
@ -116,13 +126,34 @@ class YammerProgressForm extends Form
|
|||||||
$this->out->elementEnd('fieldset');
|
$this->out->elementEnd('fieldset');
|
||||||
}
|
}
|
||||||
|
|
||||||
private function progressBar($state, $class, $label, $status)
|
private function progressBar($state, $class, $label, $status, $error=null)
|
||||||
{
|
{
|
||||||
// @fixme prettify ;)
|
// @fixme prettify ;)
|
||||||
$this->out->elementStart('div', array('class' => "import-step import-step-$state $class"));
|
$this->out->elementStart('div', array('class' => "import-step import-step-$state $class"));
|
||||||
$this->out->element('div', array('class' => 'import-label'), $label);
|
$this->out->element('div', array('class' => 'import-label'), $label);
|
||||||
$this->out->element('div', array('class' => 'import-status'), $status);
|
$this->out->element('div', array('class' => 'import-status'), $status);
|
||||||
|
if ($class == 'progress') {
|
||||||
|
if ($state == 'done') {
|
||||||
|
$this->out->submit('abort-import', _m('Reset import state'));
|
||||||
|
} else {
|
||||||
|
if ($error) {
|
||||||
|
$this->errorBox($error);
|
||||||
|
} else {
|
||||||
|
$this->out->submit('pause-import', _m('Pause import'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
$this->out->elementEnd('div');
|
$this->out->elementEnd('div');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function errorBox($msg)
|
||||||
|
{
|
||||||
|
$errline = sprintf(_m('Encountered error "%s"'), $msg);
|
||||||
|
$this->out->elementStart('fieldset', array('class' => 'import-error'));
|
||||||
|
$this->out->element('legend', array(), _m('Paused'));
|
||||||
|
$this->out->element('p', array(), $errline);
|
||||||
|
$this->out->submit('continue-import', _m('Continue'));
|
||||||
|
$this->out->submit('abort-import', _m('Abort import'));
|
||||||
|
$this->out->elementEnd('fieldset');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -38,21 +38,24 @@ class YammerQueueHandler extends QueueHandler
|
|||||||
{
|
{
|
||||||
$runner = YammerRunner::init();
|
$runner = YammerRunner::init();
|
||||||
if ($runner->hasWork()) {
|
if ($runner->hasWork()) {
|
||||||
|
try {
|
||||||
if ($runner->iterate()) {
|
if ($runner->iterate()) {
|
||||||
if ($runner->hasWork()) {
|
if ($runner->hasWork()) {
|
||||||
// More to do? Shove us back on the queue...
|
// More to do? Shove us back on the queue...
|
||||||
$runner->startBackgroundImport();
|
$runner->startBackgroundImport();
|
||||||
}
|
}
|
||||||
return true;
|
}
|
||||||
} else {
|
} catch (Exception $e) {
|
||||||
// Something failed?
|
try {
|
||||||
// @fixme should we be trying again here, or should we give warning?
|
$runner->recordError($e->getMessage());
|
||||||
return false;
|
} catch (Exception $f) {
|
||||||
|
common_log(LOG_ERR, "Error while recording error in Yammer background import: " . $e->getMessage() . " " . $f->getMessage());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// We're done!
|
// We're done!
|
||||||
common_log(LOG_INFO, "Yammer import has no work to do at this time; discarding.");
|
common_log(LOG_INFO, "Yammer import has no work to do at this time; discarding.");
|
||||||
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -298,7 +298,10 @@ class YammerRunner
|
|||||||
$this->state->state = 'save-messages';
|
$this->state->state = 'save-messages';
|
||||||
} else {
|
} else {
|
||||||
foreach ($messages as $item) {
|
foreach ($messages as $item) {
|
||||||
|
$stub = Yammer_notice_stub::staticGet($item['id']);
|
||||||
|
if (!$stub) {
|
||||||
Yammer_notice_stub::record($item['id'], $item);
|
Yammer_notice_stub::record($item['id'], $item);
|
||||||
|
}
|
||||||
$oldest = $item['id'];
|
$oldest = $item['id'];
|
||||||
}
|
}
|
||||||
$this->state->messages_oldest = $oldest;
|
$this->state->messages_oldest = $oldest;
|
||||||
@ -395,4 +398,41 @@ class YammerRunner
|
|||||||
$qm->enqueue('YammerImport', 'yammer');
|
$qm->enqueue('YammerImport', 'yammer');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Record an error condition from a background run, which we should
|
||||||
|
* display in progress state for the admin.
|
||||||
|
*
|
||||||
|
* @param string $msg
|
||||||
|
*/
|
||||||
|
public function recordError($msg)
|
||||||
|
{
|
||||||
|
// HACK HACK HACK
|
||||||
|
try {
|
||||||
|
$temp = new Yammer_state();
|
||||||
|
$temp->query('ROLLBACK');
|
||||||
|
} catch (Exception $e) {
|
||||||
|
common_log(LOG_ERR, 'Exception while confirming rollback while recording error: ' . $e->getMessage());
|
||||||
|
}
|
||||||
|
$old = clone($this->state);
|
||||||
|
$this->state->last_error = $msg;
|
||||||
|
$this->state->update($old);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Clear the error state.
|
||||||
|
*/
|
||||||
|
public function clearError()
|
||||||
|
{
|
||||||
|
$this->recordError('');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the last recorded background error message, if any.
|
||||||
|
*
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function lastError()
|
||||||
|
{
|
||||||
|
return $this->state->last_error;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user