Merge "Improve sorting on SpecialWanted*-Pages"
[mediawiki.git] / resources / src / mediawiki / mediawiki.debug.js
blob3c1a668064355b85cf443abeeee6555f613fba29
1 ( function ( mw, $ ) {
2         'use strict';
4         var debug,
5                 hovzer = $.getFootHovzer();
7         /**
8          * Debug toolbar.
9          *
10          * Enabled server-side through `$wgDebugToolbar`.
11          *
12          * @class mw.Debug
13          * @singleton
14          * @author John Du Hart
15          * @since 1.19
16          */
17         debug = mw.Debug = {
18                 /**
19                  * Toolbar container element
20                  *
21                  * @property {jQuery}
22                  */
23                 $container: null,
25                 /**
26                  * Object containing data for the debug toolbar
27                  *
28                  * @property {Object}
29                  */
30                 data: {},
32                 /**
33                  * Initialize the debugging pane
34                  *
35                  * Shouldn't be called before the document is ready
36                  * (since it binds to elements on the page).
37                  *
38                  * @param {Object} [data] Defaults to 'debugInfo' from mw.config
39                  */
40                 init: function ( data ) {
42                         this.data = data || mw.config.get( 'debugInfo' );
43                         this.buildHtml();
45                         // Insert the container into the DOM
46                         hovzer.$.append( this.$container );
47                         hovzer.update();
49                         $( '.mw-debug-panelink' ).click( this.switchPane );
50                 },
52                 /**
53                  * Switch between panes
54                  *
55                  * Should be called with an HTMLElement as its thisArg,
56                  * because it's meant to be an event handler.
57                  *
58                  * TODO: Store cookie for last pane open.
59                  *
60                  * @param {jQuery.Event} e
61                  */
62                 switchPane: function ( e ) {
63                         var currentPaneId = debug.$container.data( 'currentPane' ),
64                                 requestedPaneId = $( this ).prop( 'id' ).slice( 9 ),
65                                 $currentPane = $( '#mw-debug-pane-' + currentPaneId ),
66                                 $requestedPane = $( '#mw-debug-pane-' + requestedPaneId ),
67                                 hovDone = false;
69                         function updateHov() {
70                                 if ( !hovDone ) {
71                                         hovzer.update();
72                                         hovDone = true;
73                                 }
74                         }
76                         // Skip hash fragment handling. Prevents screen from jumping.
77                         e.preventDefault();
79                         $( this ).addClass( 'current ' );
80                         $( '.mw-debug-panelink' ).not( this ).removeClass( 'current ' );
82                         // Hide the current pane
83                         if ( requestedPaneId === currentPaneId ) {
84                                 $currentPane.slideUp( updateHov );
85                                 debug.$container.data( 'currentPane', null );
86                                 return;
87                         }
89                         debug.$container.data( 'currentPane', requestedPaneId );
91                         if ( currentPaneId === undefined || currentPaneId === null ) {
92                                 $requestedPane.slideDown( updateHov );
93                         } else {
94                                 $currentPane.hide();
95                                 $requestedPane.show();
96                                 updateHov();
97                         }
98                 },
100                 /**
101                  * Construct the HTML for the debugging toolbar
102                  */
103                 buildHtml: function () {
104                         var $container, $bits, panes, id, gitInfo;
106                         $container = $( '<div id="mw-debug-toolbar" class="mw-debug" lang="en" dir="ltr"></div>' );
108                         $bits = $( '<div class="mw-debug-bits"></div>' );
110                         /**
111                          * Returns a jQuery element for a debug-bit div
112                          *
113                          * @ignore
114                          * @param {string} id
115                          * @return {jQuery}
116                          */
117                         function bitDiv( id ) {
118                                 return $( '<div>' ).prop( {
119                                         id: 'mw-debug-' + id,
120                                         className: 'mw-debug-bit'
121                                 } )
122                                 .appendTo( $bits );
123                         }
125                         /**
126                          * Returns a jQuery element for a pane link
127                          *
128                          * @ignore
129                          * @param {string} id
130                          * @param {string} text
131                          * @return {jQuery}
132                          */
133                         function paneLabel( id, text ) {
134                                 return $( '<a>' )
135                                         .prop( {
136                                                 className: 'mw-debug-panelabel',
137                                                 href: '#mw-debug-pane-' + id
138                                         } )
139                                         .text( text );
140                         }
142                         /**
143                          * Returns a jQuery element for a debug-bit div with a for a pane link
144                          *
145                          * @ignore
146                          * @param {string} id CSS id snippet. Will be prefixed with 'mw-debug-'
147                          * @param {string} text Text to show
148                          * @param {string} count Optional count to show
149                          * @return {jQuery}
150                          */
151                         function paneTriggerBitDiv( id, text, count ) {
152                                 if ( count ) {
153                                         text = text + ' (' + count + ')';
154                                 }
155                                 return $( '<div>' ).prop( {
156                                         id: 'mw-debug-' + id,
157                                         className: 'mw-debug-bit mw-debug-panelink'
158                                 } )
159                                 .append( paneLabel( id, text ) )
160                                 .appendTo( $bits );
161                         }
163                         paneTriggerBitDiv( 'console', 'Console', this.data.log.length );
165                         paneTriggerBitDiv( 'querylist', 'Queries', this.data.queries.length );
167                         paneTriggerBitDiv( 'debuglog', 'Debug log', this.data.debugLog.length );
169                         paneTriggerBitDiv( 'request', 'Request' );
171                         paneTriggerBitDiv( 'includes', 'PHP includes', this.data.includes.length );
173                         gitInfo = '';
174                         if ( this.data.gitRevision !== false ) {
175                                 gitInfo = '(' + this.data.gitRevision.slice( 0, 7 ) + ')';
176                                 if ( this.data.gitViewUrl !== false ) {
177                                         gitInfo = $( '<a>' )
178                                                 .attr( 'href', this.data.gitViewUrl )
179                                                 .text( gitInfo );
180                                 }
181                         }
183                         bitDiv( 'mwversion' )
184                                 .append( $( '<a href="//www.mediawiki.org/">MediaWiki</a>' ) )
185                                 .append( document.createTextNode( ': ' + this.data.mwVersion + ' ' ) )
186                                 .append( gitInfo );
188                         if ( this.data.gitBranch !== false ) {
189                                 bitDiv( 'gitbranch' ).text( 'Git branch: ' + this.data.gitBranch );
190                         }
192                         bitDiv( 'phpversion' )
193                                 .append( $( this.data.phpEngine === 'HHVM' ?
194                                         '<a href="http://hhvm.com/">HHVM</a>' :
195                                         '<a href="https://php.net/">PHP</a>'
196                                 ) )
197                                 .append( ': ' + this.data.phpVersion );
199                         bitDiv( 'time' )
200                                 .text( 'Time: ' + this.data.time.toFixed( 5 ) );
202                         bitDiv( 'memory' )
203                                 .text( 'Memory: ' + this.data.memory + ' (Peak: ' + this.data.memoryPeak + ')' );
205                         $bits.appendTo( $container );
207                         panes = {
208                                 console: this.buildConsoleTable(),
209                                 querylist: this.buildQueryTable(),
210                                 debuglog: this.buildDebugLogTable(),
211                                 request: this.buildRequestPane(),
212                                 includes: this.buildIncludesPane()
213                         };
215                         for ( id in panes ) {
216                                 if ( !panes.hasOwnProperty( id ) ) {
217                                         continue;
218                                 }
220                                 $( '<div>' )
221                                         .prop( {
222                                                 className: 'mw-debug-pane',
223                                                 id: 'mw-debug-pane-' + id
224                                         } )
225                                         .append( panes[ id ] )
226                                         .appendTo( $container );
227                         }
229                         this.$container = $container;
230                 },
232                 /**
233                  * Build the console panel
234                  *
235                  * @return {jQuery} Console panel
236                  */
237                 buildConsoleTable: function () {
238                         var $table, entryTypeText, i, length, entry;
240                         $table = $( '<table id="mw-debug-console">' );
242                         $( '<colgroup>' ).css( 'width', /* padding = */ 20 + ( 10 * /* fontSize = */ 11 ) ).appendTo( $table );
243                         $( '<colgroup>' ).appendTo( $table );
244                         $( '<colgroup>' ).css( 'width', 350 ).appendTo( $table );
246                         entryTypeText = function ( entryType ) {
247                                 switch ( entryType ) {
248                                         case 'log':
249                                                 return 'Log';
250                                         case 'warn':
251                                                 return 'Warning';
252                                         case 'deprecated':
253                                                 return 'Deprecated';
254                                         default:
255                                                 return 'Unknown';
256                                 }
257                         };
259                         for ( i = 0, length = this.data.log.length; i < length; i += 1 ) {
260                                 entry = this.data.log[ i ];
261                                 entry.typeText = entryTypeText( entry.type );
263                                 $( '<tr>' )
264                                         .append( $( '<td>' )
265                                                 .text( entry.typeText )
266                                                 .addClass( 'mw-debug-console-' + entry.type )
267                                         )
268                                         .append( $( '<td>' ).html( entry.msg ) )
269                                         .append( $( '<td>' ).text( entry.caller ) )
270                                         .appendTo( $table );
271                         }
273                         return $table;
274                 },
276                 /**
277                  * Build query list pane
278                  *
279                  * @return {jQuery}
280                  */
281                 buildQueryTable: function () {
282                         var $table, i, length, query;
284                         $table = $( '<table id="mw-debug-querylist"></table>' );
286                         $( '<tr>' )
287                                 .append( $( '<th>#</th>' ).css( 'width', '4em' ) )
288                                 .append( $( '<th>SQL</th>' ) )
289                                 .append( $( '<th>Time</th>' ).css( 'width', '8em' ) )
290                                 .append( $( '<th>Call</th>' ).css( 'width', '18em' ) )
291                         .appendTo( $table );
293                         for ( i = 0, length = this.data.queries.length; i < length; i += 1 ) {
294                                 query = this.data.queries[ i ];
296                                 $( '<tr>' )
297                                         .append( $( '<td>' ).text( i + 1 ) )
298                                         .append( $( '<td>' ).text( query.sql ) )
299                                         .append( $( '<td class="stats">' ).text( ( query.time * 1000 ).toFixed( 4 ) + 'ms' ) )
300                                         .append( $( '<td>' ).text( query[ 'function' ] ) )
301                                 .appendTo( $table );
302                         }
304                         return $table;
305                 },
307                 /**
308                  * Build legacy debug log pane
309                  *
310                  * @return {jQuery}
311                  */
312                 buildDebugLogTable: function () {
313                         var $list, i, length, line;
314                         $list = $( '<ul>' );
316                         for ( i = 0, length = this.data.debugLog.length; i < length; i += 1 ) {
317                                 line = this.data.debugLog[ i ];
318                                 $( '<li>' )
319                                         .html( mw.html.escape( line ).replace( /\n/g, '<br />\n' ) )
320                                         .appendTo( $list );
321                         }
323                         return $list;
324                 },
326                 /**
327                  * Build request information pane
328                  *
329                  * @return {jQuery}
330                  */
331                 buildRequestPane: function () {
333                         function buildTable( title, data ) {
334                                 var $unit, $table, key;
336                                 $unit = $( '<div>' ).append( $( '<h2>' ).text( title ) );
338                                 $table = $( '<table>' ).appendTo( $unit );
340                                 $( '<tr>' )
341                                         .html( '<th>Key</th><th>Value</th>' )
342                                         .appendTo( $table );
344                                 for ( key in data ) {
345                                         if ( !data.hasOwnProperty( key ) ) {
346                                                 continue;
347                                         }
349                                         $( '<tr>' )
350                                                 .append( $( '<th>' ).text( key ) )
351                                                 .append( $( '<td>' ).text( data[ key ] ) )
352                                                 .appendTo( $table );
353                                 }
355                                 return $unit;
356                         }
358                         return $( '<div>' )
359                                 .text( this.data.request.method + ' ' + this.data.request.url )
360                                 .append( buildTable( 'Headers', this.data.request.headers ) )
361                                 .append( buildTable( 'Parameters', this.data.request.params ) );
362                 },
364                 /**
365                  * Build included files pane
366                  *
367                  * @return {jQuery}
368                  */
369                 buildIncludesPane: function () {
370                         var $table, i, length, file;
372                         $table = $( '<table>' );
374                         for ( i = 0, length = this.data.includes.length; i < length; i += 1 ) {
375                                 file = this.data.includes[ i ];
376                                 $( '<tr>' )
377                                         .append( $( '<td>' ).text( file.name ) )
378                                         .append( $( '<td class="nr">' ).text( file.size ) )
379                                         .appendTo( $table );
380                         }
382                         return $table;
383                 }
384         };
386         $( function () {
387                 debug.init();
388         } );
390 }( mediaWiki, jQuery ) );