SOLR-3982: Admin UI: Various Dataimport Improvements

git-svn-id: https://svn.apache.org/repos/asf/lucene/dev/trunk@1431756 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Stefan Matheis 2013-01-11 00:13:33 +00:00
parent e38d91a8f5
commit 8029a83f5a
11 changed files with 861 additions and 245 deletions

View File

@ -248,6 +248,8 @@ Optimizations
* SOLR-3840: XML query response display is unreadable in Solr Admin Query UI * SOLR-3840: XML query response display is unreadable in Solr Admin Query UI
(steffkes) (steffkes)
* SOLR-3982: Admin UI: Various Dataimport Improvements (steffkes)
Bug Fixes Bug Fixes
---------------------- ----------------------

View File

@ -29,7 +29,7 @@
#content #dataimport #form #navigation .current a #content #dataimport #form #navigation .current a
{ {
background-image: url( ../../img/ico/document-import.png ); background-image: url( ../../img/ico/status.png );
} }
#content #dataimport #form form #content #dataimport #form form
@ -60,27 +60,41 @@
width: 100%; width: 100%;
} }
#content #dataimport #form input
{
width: 98%;
}
#content #dataimport #form button #content #dataimport #form button
{ {
margin-top: 10px; margin-top: 10px;
} }
#content #dataimport #form button.loader #content #dataimport #form .execute span
{ {
background-position: 2px 50%; background-image: url( ../../img/ico/document-import.png );
padding-left: 23px; }
#content #dataimport #form .refresh-status span
{
background-image: url( ../../img/ico/arrow-circle.png );
}
#content #dataimport #form .refresh-status span.success
{
background-image: url( ../../img/ico/tick.png );
} }
#content #dataimport #form #start #content #dataimport #form #start
{ {
float: left; float: left;
margin-right: 2%; width: 47%;
width: 49%;
} }
#content #dataimport #form #rows #content #dataimport #form #rows
{ {
width: 49%; float: right;
width: 47%;
} }
#content #dataimport #form .checkbox input #content #dataimport #form .checkbox input
@ -89,9 +103,33 @@
width: auto; width: auto;
} }
#content #dataimport #form #auto-refresh-status
{
margin-top: 20px;
}
#content #dataimport #form #auto-refresh-status a
{
background-image: url( ../../img/ico/ui-check-box-uncheck.png );
background-position: 0 50%;
color: #c0c0c0;
display: block;
padding-left: 21px;
}
#content #dataimport #form #auto-refresh-status a.on,
#content #dataimport #form #auto-refresh-status a:hover
{
color: #333;
}
#content #dataimport #form #auto-refresh-status a.on
{
background-image: url( ../../img/ico/ui-check-box.png );
}
#content #dataimport #current_state #content #dataimport #current_state
{ {
display: none;
padding: 10px; padding: 10px;
margin-bottom: 20px; margin-bottom: 20px;
} }
@ -117,6 +155,30 @@
#content #dataimport #current_state .info #content #dataimport #current_state .info
{ {
background-position: 0 1px; background-position: 0 1px;
position: relative;
}
#content #dataimport #current_state .info .details span
{
color: #c0c0c0;
}
#content #dataimport #current_state .info .abort-import
{
display: none;
position: absolute;
right: 0px;
top: 0px;
}
#content #dataimport #current_state .info .abort-import span
{
background-image: url( ../../img/ico/cross.png );
}
#content #dataimport #current_state .info .abort-import.success span
{
background-image: url( ../../img/ico/tick.png );
} }
#content #dataimport #current_state.indexing #content #dataimport #current_state.indexing
@ -124,6 +186,16 @@
background-color: #f9f9f9; background-color: #f9f9f9;
} }
#content #dataimport #current_state.indexing .info
{
background-image: url( ../../img/ico/hourglass.png );
}
#content #dataimport #current_state.indexing .info .abort-import
{
display: block;
}
#content #dataimport #current_state.success #content #dataimport #current_state.success
{ {
background-color: #e6f3e6; background-color: #e6f3e6;
@ -139,6 +211,21 @@
color: #080; color: #080;
} }
#content #dataimport #current_state.aborted
{
background-color: #f3e6e6;
}
#content #dataimport #current_state.aborted .info
{
background-image: url( ../../img/ico/slash.png );
}
#content #dataimport #current_state.aborted .info strong
{
color: #800;
}
#content #dataimport #current_state.failure #content #dataimport #current_state.failure
{ {
background-color: #f3e6e6; background-color: #f3e6e6;
@ -146,7 +233,7 @@
#content #dataimport #current_state.failure .info #content #dataimport #current_state.failure .info
{ {
background-image: url( ../../img/ico/slash.png ); background-image: url( ../../img/ico/cross-button.png );
} }
#content #dataimport #current_state.failure .info strong #content #dataimport #current_state.failure .info strong
@ -154,7 +241,17 @@
color: #800; color: #800;
} }
#content #dataimport #config-error #content #dataimport #current_state.idle
{
background-color: #e6e6ff;
}
#content #dataimport #current_state.idle .info
{
background-image: url( ../../img/ico/information.png );
}
#content #dataimport #error
{ {
background-color: #f00; background-color: #f00;
background-image: url( ../../img/ico/construction.png ); background-image: url( ../../img/ico/construction.png );
@ -167,37 +264,43 @@
padding-left: 35px; padding-left: 35px;
} }
#content #dataimport #config h2 #content #dataimport .block h2
{ {
border-color: #c0c0c0; border-color: #c0c0c0;
padding-left: 5px; padding-left: 5px;
position: relative; position: relative;
} }
#content #dataimport #config.hidden h2 #content #dataimport .block.hidden h2
{ {
border-color: #fafafa; border-color: #fafafa;
} }
#content #dataimport #config h2 a.toggle #content #dataimport .block h2 a.toggle
{ {
background-image: url( ../../img/ico/toggle-small.png ); background-image: url( ../../img/ico/toggle-small.png );
background-position: 0 50%; background-position: 0 50%;
padding-left: 21px; padding-left: 21px;
} }
#content #dataimport #config.hidden h2 a.toggle #content #dataimport .block.hidden h2 a.toggle
{ {
background-image: url( ../../img/ico/toggle-small-expand.png ); background-image: url( ../../img/ico/toggle-small-expand.png );
} }
#content #dataimport #config h2 a.r
{
background-position: 3px 50%;
display: block;
float: right;
margin-left: 10px;
padding-left: 24px;
padding-right: 3px;
}
#content #dataimport #config h2 a.reload_config #content #dataimport #config h2 a.reload_config
{ {
background-image: url( ../../img/ico/arrow-circle.png ); background-image: url( ../../img/ico/arrow-circle.png );
padding-left: 21px;
position: absolute;
right: 5px;
top: 5px;
} }
#content #dataimport #config h2 a.reload_config.success #content #dataimport #config h2 a.reload_config.success
@ -210,13 +313,72 @@
background-image: url( ../../img/ico/slash.png ); background-image: url( ../../img/ico/slash.png );
} }
#content #dataimport #config.hidden .content #content #dataimport #config h2 a.debug_mode
{
background-image: url( ../../img/ico/hammer.png );
color: #c0c0c0;
}
#content #dataimport #config.debug_mode h2 a.debug_mode
{
background-color: #ff0;
background-image: url( ../../img/ico/hammer-screwdriver.png );
color: #333;
}
#content #dataimport .block.hidden .content
{ {
display: none; display: none;
} }
#content #dataimport #config .content
{
padding: 5px 2px;
}
#content #dataimport #dataimport_config .loader #content #dataimport #dataimport_config .loader
{ {
background-position: 0 50%; background-position: 0 50%;
padding-left: 21px; padding-left: 21px;
}
#content #dataimport #dataimport_config .formatted
{
border: 1px solid #fff;
display: block;
padding: 2px;
}
#content #dataimport .debug_mode #dataimport_config .formatted
{
display: none;
}
#content #dataimport #dataimport_config .editable
{
display: none;
}
#content #dataimport .debug_mode #dataimport_config .editable
{
display: block;
}
#content #dataimport #dataimport_config .editable textarea
{
font-family: monospace;
height: 120px;
min-height: 60px;
width: 100%;
}
#content #dataimport #debug_response
{
display: none;
}
#content #dataimport #debug_response em
{
color: #c0c0c0;
font-style: normal;
} }

Binary file not shown.

After

Width:  |  Height:  |  Size: 588 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 554 B

View File

@ -0,0 +1,132 @@
/*
* Auto Expanding Text Area (1.2.2)
* by Chrys Bader (www.chrysbader.com)
* chrysb@gmail.com
*
* Special thanks to:
* Jake Chapa - jake@hybridstudio.com
* John Resig - jeresig@gmail.com
*
* Copyright (c) 2008 Chrys Bader (www.chrysbader.com)
* Licensed under the GPL (GPL-LICENSE.txt) license.
*
*
* NOTE: This script requires jQuery to work. Download jQuery at www.jquery.com
*
*/
(function(jQuery) {
var self = null;
jQuery.fn.autogrow = function(o)
{
return this.each(function() {
new jQuery.autogrow(this, o);
});
};
/**
* The autogrow object.
*
* @constructor
* @name jQuery.autogrow
* @param Object e The textarea to create the autogrow for.
* @param Hash o A set of key/value pairs to set as configuration properties.
* @cat Plugins/autogrow
*/
jQuery.autogrow = function (e, o)
{
this.options = o || {};
this.dummy = null;
this.interval = null;
this.line_height = this.options.lineHeight || parseInt(jQuery(e).css('line-height'), 10);
this.min_height = this.options.minHeight || parseInt(jQuery(e).css('min-height'), 10);
this.max_height = this.options.maxHeight || parseInt(jQuery(e).css('max-height'), 10);
this.textarea = jQuery(e);
if(this.line_height == NaN)
this.line_height = 0;
// Only one textarea activated at a time, the one being used
this.init();
};
jQuery.autogrow.fn = jQuery.autogrow.prototype = {
autogrow: '1.2.2'
};
jQuery.autogrow.fn.extend = jQuery.autogrow.extend = jQuery.extend;
jQuery.autogrow.fn.extend({
init: function() {
var self = this;
this.textarea.css({overflow: 'hidden', display: 'block'});
this.textarea.bind('focus', function() { self.startExpand(); } ).bind('blur', function() { self.stopExpand(); });
this.checkExpand();
},
startExpand: function() {
var self = this;
this.interval = window.setInterval(function() {self.checkExpand();}, 400);
},
stopExpand: function() {
clearInterval(this.interval);
},
checkExpand: function() {
if (this.dummy == null)
{
this.dummy = jQuery('<div></div>');
this.dummy.css({
'font-size' : this.textarea.css('font-size'),
'font-family': this.textarea.css('font-family'),
'width' : this.textarea.css('width'),
// IE shits its pants if you uncomment the next line!
//'padding' : this.textarea.css('padding'),
'line-height': this.line_height + 'px',
'overflow-x' : 'hidden',
'position' : 'absolute',
'top' : 0,
'left' : -9999
}).appendTo('body');
}
// Strip HTML tags
var html = this.textarea.val().replace(/(<|>)/g, '');
// IE is different, as per usual
if ($.browser.msie)
{
html = html.replace(/\n/g, '<BR>new');
}
else
{
html = html.replace(/\n/g, '<br>new');
}
if (this.dummy.html() != html)
{
this.dummy.html(html);
if (this.max_height > 0 && (this.dummy.height() + this.line_height > this.max_height))
{
this.textarea.css('overflow-y', 'auto');
}
else
{
this.textarea.css('overflow-y', 'hidden');
if (this.textarea.height() < this.dummy.height() + this.line_height || (this.dummy.height() < this.textarea.height()))
{
this.textarea.animate({height: (this.dummy.height() + this.line_height) + 'px'}, 100);
}
}
}
}
});
})(jQuery);

View File

@ -20,6 +20,7 @@ require
[ [
'lib/order!lib/console', 'lib/order!lib/console',
'lib/order!jquery', 'lib/order!jquery',
'lib/order!lib/jquery.autogrow',
'lib/order!lib/jquery.cookie', 'lib/order!lib/jquery.cookie',
'lib/order!lib/jquery.form', 'lib/order!lib/jquery.form',
'lib/order!lib/jquery.jstree', 'lib/order!lib/jquery.jstree',

View File

@ -82,11 +82,7 @@ var sammy = $.sammy
{}, {},
function( context ) function( context )
{ {
if( app.timeout ) app.clear_timeout();
{
console.debug( 'Clearing Timeout #' + app.timeout );
clearTimeout( app.timeout );
}
var menu_wrapper = $( '#menu-wrapper' ); var menu_wrapper = $( '#menu-wrapper' );
@ -399,7 +395,78 @@ var solr_admin = function( app_config )
} }
} }
); );
} };
this.convert_duration_to_seconds = function convert_duration_to_seconds( str )
{
var seconds = 0;
var arr = new String( str || '' ).split( '.' );
var parts = arr[0].split( ':' ).reverse();
var parts_count = parts.length;
for( var i = 0; i < parts_count; i++ )
{
seconds += ( parseInt( parts[i], 10 ) || 0 ) * Math.pow( 60, i );
}
// treat more or equal than .5 as additional second
if( arr[1] && 5 <= parseInt( arr[1][0], 10 ) )
{
seconds++;
}
return seconds;
};
this.convert_seconds_to_readable_time = function convert_seconds_to_readable_time( seconds )
{
seconds = parseInt( seconds || 0, 10 );
var minutes = Math.floor( seconds / 60 );
var hours = Math.floor( minutes / 60 );
var text = [];
if( 0 !== hours )
{
text.push( hours + 'h' );
seconds -= hours * 60 * 60;
minutes -= hours * 60;
}
if( 0 !== minutes )
{
text.push( minutes + 'm' );
seconds -= minutes * 60;
}
if( 0 !== seconds )
{
text.push( ( '0' + seconds ).substr( -2 ) + 's' );
}
return text.join( ' ' );
};
this.clear_timeout = function clear_timeout()
{
if( !app.timeout )
{
return false;
}
console.debug( 'Clearing Timeout #' + this.timeout );
clearTimeout( this.timeout );
this.timeout = null;
};
this.format_json = function format_json( json_str )
{
if( JSON.stringify && JSON.parse )
{
json_str = JSON.stringify( JSON.parse( json_str ), undefined, 2 );
}
return json_str;
};
}; };

View File

@ -15,48 +15,8 @@
limitations under the License. limitations under the License.
*/ */
var convert_duration_to_seconds = function( str ) var dataimport_timeout = 2000;
{ var cookie_dataimport_autorefresh = 'dataimport_autorefresh';
var ret = 0;
var parts = new String( str ).split( '.' ).shift().split( ':' ).reverse();
var parts_count = parts.length;
for( var i = 0; i < parts_count; i++ )
{
ret += parseInt( parts[i], 10 ) * Math.pow( 60, i );
}
return ret;
}
var convert_seconds_to_readable_time = function( value )
{
var text = [];
value = parseInt( value );
var minutes = Math.floor( value / 60 );
var hours = Math.floor( minutes / 60 );
if( 0 !== hours )
{
text.push( hours + 'h' );
value -= hours * 60 * 60;
minutes -= hours * 60;
}
if( 0 !== minutes )
{
text.push( minutes + 'm' );
value -= minutes * 60;
}
if( 0 !== value )
{
text.push( value + 's' );
}
return text.join( ' ' );
}
sammy.bind sammy.bind
( (
@ -152,7 +112,11 @@ sammy.get
var dataimport_element = $( '#dataimport', content_element ); var dataimport_element = $( '#dataimport', content_element );
var form_element = $( '#form', dataimport_element ); var form_element = $( '#form', dataimport_element );
var config_element = $( '#config', dataimport_element ); var config_element = $( '#config', dataimport_element );
var config_error_element = $( '#config-error', dataimport_element ); var error_element = $( '#error', dataimport_element );
var debug_response_element = $( '#debug_response', dataimport_element );
var autorefresh_status = false;
var debug_mode = false;
// handler // handler
@ -195,24 +159,23 @@ sammy.get
$.ajax $.ajax
( (
{ {
url : handler_url + '?command=show-config', url : handler_url + '?command=show-config&indent=true',
dataType : 'xml', dataType : 'xml',
context : $( '#dataimport_config', config_element ), context : $( '#dataimport_config', config_element ),
beforeSend : function( xhr, settings ) beforeSend : function( xhr, settings )
{ {
error_element
.empty()
.hide();
}, },
success : function( config, text_status, xhr ) success : function( config, text_status, xhr )
{ {
dataimport_element dataimport_element
.removeClass( 'error' ); .removeClass( 'error' );
config_error_element
.hide();
config_element config_element
.addClass( 'hidden' ); .addClass( 'hidden' );
var entities = [ '<option value=""></option>' ]; var entities = [ '<option value=""></option>' ];
$( 'document > entity', config ) $( 'document > entity', config )
@ -226,6 +189,9 @@ sammy.get
$( '#entity', form_element ) $( '#entity', form_element )
.html( entities.join( "\n" ) ); .html( entities.join( "\n" ) );
$( '.editable textarea', this )
.val( xhr.responseText.replace( /\n+$/, '' ) );
}, },
error : function( xhr, text_status, error_thrown ) error : function( xhr, text_status, error_thrown )
{ {
@ -234,7 +200,8 @@ sammy.get
dataimport_element dataimport_element
.addClass( 'error' ); .addClass( 'error' );
config_error_element error_element
.text( 'Dataimport XML-Configuration is not valid' )
.show(); .show();
config_element config_element
@ -248,7 +215,7 @@ sammy.get
xhr.responseText.esc() + xhr.responseText.esc() +
'</code></pre>' '</code></pre>'
); );
this.html( code ); $( '.formatted', this ).html( code );
if( 'success' === text_status ) if( 'success' === text_status )
{ {
@ -260,7 +227,7 @@ sammy.get
} }
dataimport_fetch_config(); dataimport_fetch_config();
$( '.toggle', config_element ) $( '.block .toggle', dataimport_element )
.die( 'click' ) .die( 'click' )
.live .live
( (
@ -325,160 +292,339 @@ sammy.get
); );
return false; return false;
} }
) );
var debug_mode_element = $( '.debug_mode', config_element );
debug_mode_element
.die( 'click' )
.live
(
'click',
function( event )
{
var self = $( this );
var block = self.closest( '.block' )
var debug_checkbox = $( 'input[name="debug"]', form_element );
var submit_span = $( 'button[type="submit"] span', form_element );
debug_mode = !debug_mode;
block.toggleClass( 'debug_mode', debug_mode );
if( debug_mode )
{
block.removeClass( 'hidden' );
debug_checkbox
.attr( 'checked', 'checked' )
.trigger( 'change' );
submit_span
.data( 'original', submit_span.text() )
.text( submit_span.data( 'debugmode' ) );
$( 'textarea', block )
.autogrow()
}
else
{
submit_span
.text( submit_span.data( 'original' ) )
.removeData( 'original' );
}
}
);
// abort
var abort_import_element = $( '.abort-import', dataimport_element );
abort_import_element
.off( 'click' )
.on
(
'click',
function( event )
{
var span_element = $( 'span', this );
$.ajax
(
{
url : handler_url + '?command=abort&wt=json',
dataType : 'json',
type: 'POST',
context: $( this ),
beforeSend : function( xhr, settings )
{
span_element
.addClass( 'loader' );
},
success : function( response, text_status, xhr )
{
span_element
.data( 'original', span_element.text() )
.text( span_element.data( 'aborting' ) );
this
.removeClass( 'warn' )
.addClass( 'success' );
window.setTimeout
(
function()
{
$( 'span', abort_import_element )
.removeClass( 'loader' )
.text( span_element.data( 'original' ) )
.removeData( 'original' );
abort_import_element
.removeClass( 'success' )
.addClass( 'warn' );
},
dataimport_timeout * 2
);
dataimport_fetch_status();
}
}
);
return false;
}
);
// state // state
var status_button = $( 'form button.refresh-status', form_element );
status_button
.off( 'click' )
.on
(
'click',
function( event )
{
dataimport_fetch_status();
return false;
}
)
.trigger( 'click' );
function dataimport_fetch_status() function dataimport_fetch_status( clear_timeout )
{ {
if( clear_timeout )
{
app.clear_timeout();
}
$.ajax $.ajax
( (
{ {
url : handler_url + '?command=status', url : handler_url + '?command=status&indent=true&wt=json',
dataType : 'xml', dataType : 'json',
beforeSend : function( xhr, settings ) beforeSend : function( xhr, settings )
{ {
$( 'span', status_button )
.addClass( 'loader' );
}, },
success : function( response, text_status, xhr ) success : function( response, text_status, xhr )
{ {
var state_element = $( '#current_state', content_element ); var state_element = $( '#current_state', content_element );
var time_element = $( '.time', state_element );
var status = $( 'str[name="status"]', response ).text(); var status = response.status;
var rollback_element = $( 'str[name="Rolledback"]', response ); var rollback_time = response.statusMessages.Rolledback || null;
var messages_count = $( 'lst[name="statusMessages"] str', response ).size(); var abort_time = response.statusMessages.Aborted || null;
var messages = response.statusMessages;
var messages_count = 0;
for( var key in messages ) { messages_count++; }
var started_at = $( 'str[name="Full Dump Started"]', response ).text(); var format_number = function format_number( number )
if( !started_at )
{ {
started_at = (new Date()).toGMTString(); return ( number || 0 ).toString().replace( /\B(?=(\d{3})+(?!\d))/g, '\'' );
} };
function dataimport_compute_details( response, details_element ) function dataimport_compute_details( response, details_element, elapsed_seconds )
{ {
var config = {
'Requests' : 'Total Requests made to DataSource',
'Fetched' : 'Total Rows Fetched',
'Skipped' : 'Total Documents Skipped',
'Processed' : 'Total Documents Processed'
};
var details = []; var details = [];
for( var key in config )
var requests = parseInt( $( 'str[name="Total Requests made to DataSource"]', response ).text(), 10 );
if( requests )
{ {
details.push var value = parseInt( response.statusMessages[config[key]], 10 );
(
'<abbr title="Total Requests made to DataSource">Requests</abbr>: ' +
requests
);
}
var fetched = parseInt( $( 'str[name="Total Rows Fetched"]', response ).text(), 10 ); var detail = '<abbr title="' + config[key].esc() + '">' + key.esc() + '</abbr>: ' + format_number( value ).esc();
if( fetched ) if( elapsed_seconds && 'skipped' !== key.toLowerCase() )
{ {
details.push detail += ' <span>(' + format_number( Math.round( value / elapsed_seconds ) ).esc() + '/s)</span>'
( }
'<abbr title="Total Rows Fetched">Fetched</abbr>: ' +
fetched
);
}
var skipped = parseInt( $( 'str[name="Total Documents Skipped"]', response ).text(), 10 ); details.push( detail );
if( requests ) };
{
details.push
(
'<abbr title="Total Documents Skipped">Skipped</abbr>: ' +
skipped
);
}
var processed = parseInt( $( 'str[name="Total Documents Processed"]', response ).text(), 10 );
if( processed )
{
details.push
(
'<abbr title="Total Documents Processed">Processed</abbr>: ' +
processed
);
}
details_element details_element
.html( details.join( ', ' ) ) .html( details.join( ', ' ) )
.show(); .show();
};
var get_time_taken = function get_default_time_taken()
{
var time_taken_text = response.statusMessages['Time taken'];
return app.convert_duration_to_seconds( time_taken_text );
};
var get_default_info_text = function default_info_text()
{
var info_text = response.statusMessages[''] || '';
// format numbers included in status nicely
info_text = info_text.replace
(
/\d{4,}/g,
function( match, position, string )
{
return format_number( parseInt( match, 10 ) );
}
);
var time_taken_text = app.convert_seconds_to_readable_time( get_time_taken() );
if( time_taken_text )
{
info_text += ' (Duration: ' + time_taken_text.esc() + ')';
}
return info_text;
};
var show_info = function show_info( info_text, elapsed_seconds )
{
$( '.info strong', state_element )
.text( info_text || get_default_info_text() );
$( '.info .details', state_element )
.hide();
};
var show_full_info = function show_full_info( info_text, elapsed_seconds )
{
show_info( info_text, elapsed_seconds );
dataimport_compute_details
(
response,
$( '.info .details', state_element ),
elapsed_seconds || get_time_taken()
);
};
var set_time = function set_time( time_text )
{
time_element
.text( time_text )
.removeData( 'timeago' )
.timeago()
.show();
} }
state_element state_element
.removeClass( 'indexing' ) .removeAttr( 'class' );
.removeClass( 'success' )
.removeClass( 'failure' ); time_element
.empty()
.hide();
$( '.info', state_element ) $( '.info', state_element )
.removeClass( 'loader' ); .removeClass( 'loader' );
if( 0 !== rollback_element.size() ) if( 'busy' === status )
{ {
state_element state_element
.addClass( 'failure' ) .addClass( 'indexing' );
.show();
$( '.time', state_element ) if( autorefresh_status )
.text( rollback_element.text() ) {
.timeago() $( '.info', state_element )
.show(); .addClass( 'loader' );
}
$( '.info strong', state_element ) var time_elapsed_text = response.statusMessages['Time Elapsed'];
.text( $( 'str[name=""]', response ).text() ); var elapsed_seconds = app.convert_duration_to_seconds( time_elapsed_text );
time_elapsed_text = app.convert_seconds_to_readable_time( elapsed_seconds );
$( '.info .details', state_element ) var info_text = time_elapsed_text
.hide(); ? 'Indexing since ' + time_elapsed_text
: 'Indexing ...';
console.debug( 'rollback @ ', rollback_element.text() );
show_full_info( info_text, elapsed_seconds );
if( !app.timeout && autorefresh_status )
{
app.timeout = window.setTimeout
(
function()
{
dataimport_fetch_status( true )
},
dataimport_timeout
);
}
}
else if( rollback_time )
{
state_element
.addClass( 'failure' );
set_time( rollback_time );
show_full_info();
}
else if( abort_time )
{
state_element
.addClass( 'aborted' );
set_time( abort_time );
show_full_info( 'Aborting current Import ...' );
} }
else if( 'idle' === status && 0 !== messages_count ) else if( 'idle' === status && 0 !== messages_count )
{ {
state_element state_element
.addClass( 'success' ) .addClass( 'success' );
.show();
$( '.time', state_element ) var started_at = response.statusMessages['Full Dump Started'];
.text( started_at ) if( started_at )
.timeago()
.show();
$( '.info strong', state_element )
.text( $( 'str[name=""]', response ).text() );
dataimport_compute_details( response, $( '.info .details', state_element ) );
}
else if( 'busy' === status )
{
state_element
.addClass( 'indexing' )
.show();
$( '.time', state_element )
.text( started_at )
.timeago()
.show();
$( '.info', state_element )
.addClass( 'loader' );
var indexing_text = 'Indexing ...';
var time_elapsed_text = $( 'str[name="Time Elapsed"]', response ).text();
time_elapsed_text = convert_seconds_to_readable_time( convert_duration_to_seconds( time_elapsed_text ) );
if( time_elapsed_text.length )
{ {
indexing_text = 'Indexing since ' + time_elapsed_text set_time( started_at );
} }
$( '.info strong', state_element ) show_full_info();
.text( indexing_text );
dataimport_compute_details( response, $( '.info .details', state_element ) );
window.setTimeout( dataimport_fetch_status, 2000 );
} }
else else
{ {
state_element.hide(); state_element
.addClass( 'idle' );
show_info( 'No information available (idle)' );
} }
// show raw status
var code = $(
'<pre class="syntax language-json"><code>' +
app.format_json( xhr.responseText ).esc() +
'</code></pre>'
);
$( '#raw_output_container', content_element ).html( code );
hljs.highlightBlock( code.get(0) );
}, },
error : function( xhr, text_status, error_thrown ) error : function( xhr, text_status, error_thrown )
{ {
@ -489,24 +635,47 @@ sammy.get
}, },
complete : function( xhr, text_status ) complete : function( xhr, text_status )
{ {
$( 'span', status_button )
.removeClass( 'loader' )
.addClass( 'success' );
window.setTimeout
(
function()
{
$( 'span', status_button )
.removeClass( 'success' );
},
dataimport_timeout / 2
);
} }
} }
); );
} }
dataimport_fetch_status();
// form // form
$( 'form', form_element ) var form = $( 'form', form_element );
form
.ajaxForm .ajaxForm
( (
{ {
url : handler_url, url : handler_url,
dataType : 'xml', data : {
wt : 'json',
indent : 'true'
},
dataType : 'json',
type: 'POST',
beforeSend : function( xhr, settings ) beforeSend : function( xhr, settings )
{ {
$( 'form button', form_element ) $( 'button[type="submit"] span', form_element )
.addClass( 'loader' ); .addClass( 'loader' );
error_element
.empty()
.hide();
}, },
beforeSubmit : function( array, form, options ) beforeSubmit : function( array, form, options )
{ {
@ -545,22 +714,91 @@ sammy.get
array.push( { name : tmp[0], value: tmp[1] } ); array.push( { name : tmp[0], value: tmp[1] } );
} }
} }
if( debug_mode )
{
array.push( { name: 'dataConfig', value: $( '#dataimport_config .editable textarea' ).val() } );
}
}, },
success : function( response, text_status, xhr ) success : function( response, text_status, xhr )
{ {
dataimport_fetch_status();
}, },
error : function( xhr, text_status, error_thrown ) error : function( xhr, text_status, error_thrown )
{ {
console.debug( arguments ); var response = null;
try
{
eval( 'response = ' + xhr.responseText + ';' );
}
catch( e ){}
error_element
.text( response.error.msg || 'Unknown Error (Exception w/o Message)' )
.show();
}, },
complete : function( xhr, text_status ) complete : function( xhr, text_status )
{ {
$( 'form button', form_element ) $( 'button[type="submit"] span', form_element )
.removeClass( 'loader' ); .removeClass( 'loader' );
var debug = $( 'input[name="debug"]:checked', form );
if( 0 !== debug.size() )
{
var code = $(
'<pre class="syntax language-json"><code>' +
app.format_json( xhr.responseText ).esc() +
'</code></pre>'
);
$( '.content', debug_response_element ).html( code );
hljs.highlightBlock( code.get(0) );
}
dataimport_fetch_status();
} }
} }
); );
$( 'input[name="debug"]', form )
.off( 'change' )
.on
(
'change',
function( event )
{
debug_response_element.toggle( this.checked );
}
);
$( '#auto-refresh-status a', form_element )
.off( 'click' )
.on
(
'click',
function( event )
{
$.cookie( cookie_dataimport_autorefresh, $.cookie( cookie_dataimport_autorefresh ) ? null : true );
$( this ).trigger( 'state' );
dataimport_fetch_status();
return false;
}
)
.off( 'state' )
.on
(
'state',
function( event )
{
autorefresh_status = !!$.cookie( cookie_dataimport_autorefresh );
$.cookie( cookie_dataimport_autorefresh )
? $( this ).addClass( 'on' )
: $( this ).removeClass( 'on' );
}
)
.trigger( 'state' );
} }
); );
} }

View File

@ -56,7 +56,7 @@ sammy.get
json : function( xhr ) json : function( xhr )
{ {
return JSON.stringify( JSON.parse( xhr.responseText ), undefined, 2 ); return app.format_json( xhr.responseText );
} }
}; };

View File

@ -22,49 +22,6 @@ var core_basepath = null;
var navigation_element = null; var navigation_element = null;
var replication_element = null; var replication_element = null;
var convert_duration_to_seconds = function( str )
{
var ret = 0;
var parts = new String( str ).split( ':' ).reverse();
var parts_count = parts.length;
for( var i = 0; i < parts_count; i++ )
{
ret += parseInt( parts[i], 10 ) * Math.pow( 60, i );
}
return ret;
}
var convert_seconds_to_readable_time = function( value )
{
var text = [];
value = parseInt( value );
var minutes = Math.floor( value / 60 );
var hours = Math.floor( minutes / 60 );
if( 0 !== hours )
{
text.push( hours + 'h' );
value -= hours * 60 * 60;
minutes -= hours * 60;
}
if( 0 !== minutes )
{
text.push( minutes + 'm' );
value -= minutes * 60;
}
if( 0 !== value )
{
text.push( value + 's' );
}
return text.join( ' ' );
}
var init_timer = function( next_tick ) var init_timer = function( next_tick )
{ {
if( timer_timeout ) if( timer_timeout )
@ -83,7 +40,7 @@ var update_timer = function( next_tick )
} }
$( 'p .tick', timer_element ) $( 'p .tick', timer_element )
.text( convert_seconds_to_readable_time( next_tick ) ); .text( app.convert_seconds_to_readable_time( next_tick ) );
timer_timeout = window.setTimeout timer_timeout = window.setTimeout
( (
@ -151,7 +108,7 @@ var replication_fetch_status = function()
var eta_element = $( '#eta', progress_element ); var eta_element = $( '#eta', progress_element );
$( 'span', eta_element ) $( 'span', eta_element )
.text( convert_seconds_to_readable_time( data.slave.timeRemaining ) ); .text( app.convert_seconds_to_readable_time( data.slave.timeRemaining ) );
var bar_element = $( '#bar', progress_element ); var bar_element = $( '#bar', progress_element );
$( '.files span', bar_element ) $( '.files span', bar_element )
@ -394,7 +351,7 @@ var replication_fetch_status = function()
timer_element = $( '.timer', navigation_element ); timer_element = $( '.timer', navigation_element );
approx_element = $( '.approx', timer_element ); approx_element = $( '.approx', timer_element );
var next_tick = convert_duration_to_seconds( data.slave.pollInterval ); var next_tick = app.convert_duration_to_seconds( data.slave.pollInterval );
approx_element.show(); approx_element.show();
if( data.slave.nextExecutionAt ) if( data.slave.nextExecutionAt )

View File

@ -18,27 +18,46 @@ limitations under the License.
<div id="frame"> <div id="frame">
<div id="error"></div>
<div id="current_state"> <div id="current_state">
<span class="time"></span> <span class="time"></span>
<div class="info"> <div class="info">
<strong></strong> <strong></strong>
<div class="details"></div> <div class="details"></div>
<button class="abort-import warn"><span data-aborting="Aborting Import">Abort Import</span></button>
</div> </div>
</div> </div>
<div id="config-error"> <div class="block hidden" id="raw_output">
Dataimport XML-Configuration is not valid <h2>
<a class="toggle"><span>Raw Status-Output</span></a>
</h2>
<div class="message-container">
<div class="message"></div>
</div>
<div class="content">
<div id="raw_output_container"></div>
</div>
</div> </div>
<div class="block hidden" id="config"> <div class="block hidden" id="config">
<h2> <h2 class="clearfix">
<a class="toggle"><span>Dataimport Configuration</span></a> <a class="toggle"><span>Configuration</span></a>
<a class="reload_config" title="Reload Configuration">Reload</a> <a class="r reload_config" title="Reload Configuration">Reload</a>
<a class="r debug_mode">Debug-Mode</a>
</h2> </h2>
<div class="message-container"> <div class="message-container">
@ -49,13 +68,41 @@ limitations under the License.
<div id="dataimport_config"> <div id="dataimport_config">
<div class="loader">Loading ...</div> <div class="formatted">
<div class="loader">Loading ...</div>
</div>
<div class="editable">
<textarea></textarea>
</div>
</div> </div>
</div> </div>
</div> </div>
<div class="block hidden" id="debug_response">
<h2>
<a class="toggle"><span>Raw Debug-Response</span></a>
</h2>
<div class="message-container">
<div class="message"></div>
</div>
<div class="content">
<em>No Request executed</em>
</div>
</div>
</div> </div>
@ -97,6 +144,11 @@ limitations under the License.
<a rel="help">Optimize</a> <a rel="help">Optimize</a>
</label> </label>
<label for="debug" class="checkbox">
<input type="checkbox" name="debug" id="debug" value="true">
<a rel="help">Debug</a>
</label>
<label for="entity"> <label for="entity">
<a rel="help">Entity</a> <a rel="help">Entity</a>
</label> </label>
@ -106,17 +158,22 @@ limitations under the License.
<a rel="help">Start</a>, <a rel="help">Start</a>,
<a rel="help">Rows</a> <a rel="help">Rows</a>
</label> </label>
<input type="text" id="start" placeholder="0"> <div class="clearfix">
<input type="text" id="rows" placeholder="10"> <input type="text" id="start" placeholder="0">
<input type="text" id="rows" placeholder="10">
</div>
<label for="custom_parameters"> <label for="custom_parameters">
<a rel="help">Custom Parameters</a> <a rel="help">Custom Parameters</a>
</label> </label>
<input type="text" id="custom_parameters" value="" placeholder="key1=val1&amp;key2=val2"> <input type="text" id="custom_parameters" value="" placeholder="key1=val1&amp;key2=val2">
<button type="submit">Execute Import</button> <button class="execute" type="submit"><span data-debugmode="Execute with this Configuration →">Execute</span></button>
<button class="refresh-status"><span>Refresh Status</span></button>
</form> </form>
<p id="auto-refresh-status"><a>Auto-Refresh Status</a></p>
</div> </div>