| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387 | 
							- // Copyright 2012 United States Government, as represented by the Secretary of Defense, Under
 
- // Secretary of Defense (Personnel & Readiness).
 
- // 
 
- // Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
 
- // in compliance with the License. You may obtain a copy of the License at
 
- // 
 
- //   http://www.apache.org/licenses/LICENSE-2.0
 
- // 
 
- // Unless required by applicable law or agreed to in writing, software distributed under the License
 
- // is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
 
- // or implied. See the License for the specific language governing permissions and limitations under
 
- // the License.
 
- /// @module logger
 
- /// @requires vwf/configuration
 
- define( [ "vwf/configuration" ], function( configuration ) {
 
-     var TRACE = 1,
 
-         DEBUG = 2,
 
-         INFO = 3,
 
-         WARN = 4,
 
-         ERROR = 5,
 
-         FATAL = 6;
 
-     var exports = {
 
-         levels: {
 
-             trace: TRACE,
 
-             debug: DEBUG,
 
-             info: INFO,
 
-             warn: WARN,
 
-             error: ERROR,
 
-             // fatal: FATAL,
 
-         },
 
-         label: undefined,
 
-         context: undefined,
 
-         level: WARN,
 
-         for: function( label, context, level ) {
 
-             return Object.create( this ).configure( label, context, level );
 
-         },
 
-         configure: function( label, context, level ) {
 
-             var proto = Object.getPrototypeOf( this ) !== Object.prototype ?
 
-                 Object.getPrototypeOf( this ) : undefined;
 
-             this.label = combined_label( proto && proto.label, label );
 
-             this.context = context || proto && proto.context;
 
-             this.level = { name: "warn", number: WARN }; // default
 
-             switch( level ) {
 
-                 case "trace": case "TRACE": this.level = { name: "trace", number: TRACE }; break;
 
-                 case "debug": case "DEBUG": this.level = { name: "debug", number: DEBUG }; break;
 
-                 case "info":  case "INFO":  this.level = { name: "info",  number: INFO };  break;
 
-                 case "warn":  case "WARN":  this.level = { name: "warn",  number: WARN };  break;
 
-                 case "error": case "ERROR": this.level = { name: "error", number: ERROR }; break;
 
-                 // case "fatal": case "FATAL": this.level = { name: "fatal", number: FATAL }; break;
 
-                 default: proto && delete this.level; break; // inherit from the prototype
 
-             }
 
-             return this;
 
-         },
 
-         /// Log a message and open a group if the log threshold is "trace" or below.
 
-         traceg: function( /* ... */ ) {
 
-             TRACE >= this.level.number &&
 
-                 log.call( this, arguments, console && console.group, console );
 
-         },
 
-         /// Log a message and open a group if the log threshold is "debug" or below.
 
-         debugg: function( /* ... */ ) {
 
-             DEBUG >= this.level.number &&
 
-                 log.call( this, arguments, console && console.group, console );
 
-         },
 
-         /// Log a message and open a group if the log threshold is "info" or below.
 
-         infog: function( /* ... */ ) {
 
-             INFO >= this.level.number &&
 
-                 log.call( this, arguments, console && console.group, console );
 
-         },
 
-         /// Close a group if the log threshold is "trace" or below.
 
-         traceu: function() {
 
-             TRACE >= this.level.number &&
 
-                 log.call( this, arguments, console && console.groupEnd, console );
 
-         },
 
-         /// Close a group if the log threshold is "debug" or below.
 
-         debugu: function() {
 
-             DEBUG >= this.level.number &&
 
-                 log.call( this, arguments, console && console.groupEnd, console );
 
-         },
 
-         /// Close a group if the log threshold is "info" or below.
 
-         infou: function() {
 
-             INFO >= this.level.number &&
 
-                 log.call( this, arguments, console && console.groupEnd, console );
 
-         },
 
-         /// Log a message if the log threshold is "trace" or below.
 
-         trace: function( /* ... */ ) {
 
-             TRACE >= this.level.number &&
 
-                 log.call( this, arguments, console && console.debug, console ); // not console.trace(), which would log the stack
 
-         },
 
-         /// Log a message if the log threshold is "debug" or below.
 
-         debug: function( /* ... */ ) {
 
-             DEBUG >= this.level.number &&
 
-                 log.call( this, arguments, console && console.debug, console );
 
-         },
 
-         /// Log a message if the log threshold is "info" or below.
 
-         info: function( /* ... */ ) {
 
-             INFO >= this.level.number &&
 
-                 log.call( this, arguments, console && console.info, console );
 
-         },
 
-         /// Log a message if the log threshold is "warn" or below.
 
-         warn: function( /* ... */ ) {
 
-             WARN >= this.level.number &&
 
-                 log.call( this, arguments, console && console.warn, console );
 
-         },
 
-         /// Log a message if the log threshold is "error" or below.
 
-         error: function( /* ... */ ) {
 
-             ERROR >= this.level.number &&
 
-                 log.call( this, arguments, console && console.error, console );
 
-         },
 
-         /// Log a message.
 
-         log: function( /* ... */ ) {
 
-             log.call( this, arguments, console && console.log, console );
 
-         },
 
-         // Log with an extra one-time label. Equivalent to this.logger.for( label ).log( ... ),
 
-         // etc., but without the overhead of creating a new logger.
 
-         /// Log a message with an extra one-time label and open a group if the log threshold is
 
-         /// "trace" or below.
 
-         tracegx: function( /* label, ... */ ) {
 
-             TRACE >= this.level.number &&
 
-                 log.call( this, arguments, console && console.group, console, true );
 
-         },
 
-         /// Log a message with an extra one-time label and open a group if the log threshold is
 
-         /// "debug" or below.
 
-         debuggx: function( /* label, ... */ ) {
 
-             DEBUG >= this.level.number &&
 
-                 log.call( this, arguments, console && console.group, console, true );
 
-         },
 
-         /// Log a message with an extra one-time label and open a group if the log threshold is
 
-         /// "info" or below.
 
-         infogx: function( /* label, ... */ ) {
 
-             INFO >= this.level.number &&
 
-                 log.call( this, arguments, console && console.group, console, true );
 
-         },
 
-         /// Log a message with an extra one-time label if the log threshold is "trace" or below.
 
-         tracex: function( /* ... */ ) {
 
-             TRACE >= this.level.number &&
 
-                 log.call( this, arguments, console && console.debug, console, true ); // not console.trace(), which would log the stack
 
-         },
 
-         /// Log a message with an extra one-time label if the log threshold is "debug" or below.
 
-         debugx: function( /* label, ... */ ) {
 
-             DEBUG >= this.level.number &&
 
-                 log.call( this, arguments, console && console.debug, console, true );
 
-         },
 
-         /// Log a message with an extra one-time label if the log threshold is "info" or below.
 
-         infox: function( /* label, ... */ ) {
 
-             INFO >= this.level.number &&
 
-                 log.call( this, arguments, console && console.info, console, true );
 
-         },
 
-         /// Log a message with an extra one-time label if the log threshold is "warn" or below.
 
-         warnx: function( /* label, ... */ ) {
 
-             WARN >= this.level.number &&
 
-                 log.call( this, arguments, console && console.warn, console, true );
 
-         },
 
-         /// Log a message with an extra one-time label if the log threshold is "error" or below.
 
-         errorx: function( /* label, ... */ ) {
 
-             ERROR >= this.level.number &&
 
-                 log.call( this, arguments, console && console.warn, console, true );
 
-         },
 
-         /// Log a message with an extra one-time label.
 
-         logx: function( /* label, ... */ ) {
 
-             log.call( this, arguments, console && console.log, console, true );
 
-         },
 
-     };
 
-     /// Log a message to the console. Normalize the arguments list and invoke the appender function.
 
-     /// 
 
-     /// @param {Array} args
 
-     ///   An Array-like list of arguments passed to a log function. normalize describes the formats
 
-     ///   supported.
 
-     /// @param {Function} appender
 
-     ///   A Firebug-like log function that logs its arguments, such as window.console.log.
 
-     /// @param {Object} context
 
-     ///   The *this* object for the appender, such as window.console.
 
-     /// @param {Boolean} [extra]
 
-     ///   If true, interpret args[0] as a one-time label that extends the logger's output prefix.
 
-     function log( args, appender, context, extra ) {  // invoke with *this* as the logger module
 
-         // Normalize the arguments and log the message. Don't log a message if normalize() returned
 
-         // undefined (because a generator function didn't return a result).
 
-         if ( args = /* assignment! */ normalize.call( this, args, extra ) ) {
 
-             appender && appender.apply( context, args );
 
-         }
 
-     }
 
-     /// Normalize the arguments provided to a log function. The arguments may take one of the
 
-     /// following forms:
 
-     /// 
 
-     /// A series of values, or a function that generates the values:
 
-     /// 
 
-     ///   [ value, value, ... ]
 
-     ///   [ function( name, number ) { return [ value, value, ... ] }, context ]
 
-     /// 
 
-     /// For a generator function, an optional context argument provides the function's *this*
 
-     /// context. The logger's default context will be used if the context argument is no provided.
 
-     /// The log threshold name and number ("info" and 3, for example) are passed as arguments to the
 
-     /// function.
 
-     /// 
 
-     /// No message will be logged if a generator function returns undefined. Since the function will
 
-     /// only be called if the message type exceeds the log threshold, logger calls may be used to
 
-     /// control program behavior based on the log level:
 
-     ///
 
-     ///   this.logger.debug( function( name, number ) {
 
-     ///       debugControls.visible = true; // returns undefined, so no message
 
-     ///   } );
 
-     /// 
 
-     /// When *extra* is truthy, the first argument is interpreted as a one-time label that extends
 
-     /// the logger's output prefix:
 
-     /// 
 
-     ///   [ "label", value, value, ... ]
 
-     ///   [ "label", function( name, number ) { return [ value, value, ... ] }, context ]
 
-     /// 
 
-     /// The arguments are normalized into a list of values ready to pass to the appender:
 
-     /// 
 
-     ///   [ value, value, ... ]
 
-     /// 
 
-     /// @param {Array} args
 
-     ///   An Array-like list of arguments passed to one of the log functions.
 
-     /// @param {Boolean} [extra]
 
-     ///   If true, interpret args[0] as a one-time label that extends the logger's output prefix.
 
-     function normalize( args, extra ) {  // invoke with *this* as the logger module
 
-         // Record the extra one-time label if one is provided. We leave it in the arguments list so
 
-         // that we don't convert Arguments to an Array if it isn't necessary.
 
-         if ( extra && ( typeof args[0] == "string" || args[0] instanceof String ) ) {
 
-             var label = args[0];
 
-             var start = 1;
 
-         } else {
 
-             var label = undefined;
 
-             var start = 0;
 
-         }
 
-         // If a generator function is provided (instead of a series of values), call it to get the
 
-         // arguments list.
 
-         if ( typeof args[ start ] == "function" || args[ start ] instanceof Function ) {
 
-             // Call the function using the provided context or this logger's context. We expect the
 
-             // function to return an array of values, a single value, or undefined.
 
-             args = args[ start ].call( args[ start+1 ] || this.context, this.level.name, this.level.number );
 
-             // Convert a single value to an Array. An Array remains an Array. Leave undefined
 
-             // unchanged.
 
-             if ( args !== undefined && ( typeof args != "object" || ! ( args instanceof Array ) ) ) {
 
-                 args = [ args ];
 
-             }
 
-         } else {
 
-             // Remove the extra one-time label.
 
-             if ( start > 0 ) {
 
-                 args = Array.prototype.slice.call( args, start );
 
-             }
 
-         }
 
-         // Add the prefix label to the arguments list and return. But return undefined if a
 
-         // generator didn't return a result.
 
-         return args ? prefixed_arguments( this.label, label, args ) : undefined;
 
-     }
 
-     /// Update an arguments list to prepend "<label>: " to the output.
 
-     function prefixed_arguments( label, extra, args ) {
 
-         if ( label || extra ) {
 
-             if ( args.length == 0 ) {
 
-                 return [ combined_label( label, extra ) ]; // just show the module and function name when there are no additional arguments
 
-             } else if ( typeof args[0] == "string" || args[0] instanceof String ) {
 
-                 return [ combined_label( label, extra ) + ": " + args[0] ].concat( Array.prototype.slice.call( args, 1 ) ); // concatenate when the first field is a string so that it may remain a format string
 
-             } else {
 
-                 return [ combined_label( label, extra ) + ":" ].concat( args ); // otherwise insert a new first field
 
-             }
 
-         } else {
 
-             return args;
 
-         }
 
-     }
 
-     /// Generate a new label from a parent label and an extra part.
 
-     function combined_label( label, extra ) {
 
-         // Combine with "." unless the extension provides its own separator.
 
-         var separator = extra && extra.match( /^[0-9A-Za-z]/ ) ? "." : "";
 
-         // Concatenate and return.
 
-         if ( label && extra ) {
 
-             return label + separator + extra;
 
-         } else if ( extra ) {
 
-             return extra;
 
-         } else if ( label ) {
 
-             return label;
 
-         } else {
 
-             return undefined;
 
-         }
 
-     }
 
-     // Get the initial level setting from the configuration module, and update the level when the
 
-     // configuration changes.
 
-     // TODO: should be done somewhere else; the logger isn't bound to VWF and shouldn't have a dependency on the configuration module.
 
-     exports.configure( undefined, undefined, configuration.active["log-level"] || "warn" );
 
-     configuration.changed( function( active ) {
 
-         exports.configure( undefined, undefined, active["log-level"] || "warn" );
 
-     }, this );
 
-     // Return the module.
 
-     return exports;
 
- } );
 
 
  |