@knev/bitlogr v2.0.11
BitLOGR
BitLOGR is a lightweight, bitwise logging library for JavaScript, designed for modular applications with no label dependencies between compilation units. It leverages bit flags for efficient, granular logging control, offering zero performance overhead in production through lazy evaluation (thunks) and build-time optimizations.
The idea behind BitLOGR is to label execution paths, not INFO, WARN, DEBUG, ERROR, CRITICAL.
Key Features (USPs)
- No Label Dependency: Each compilation unit can independently define, ignore, inherit, overwrite, or rename labels from submodules.
- Flexible Label Management: Supports custom label sets with bitwise operations for combining and toggling logs.
- Custom Logging Handler: It is possible to specify your own handler (e.g., console.log, custom function) for output flexibility.
- Zero Performance Hit: Use thunks, to defer argument evaluation, and static functions to, ensuring no overhead when logging is disabled.
Installation
Install via npm:
npm install @knev/bitlogrOverview
Singleton Class
BitLOGR is implemented as a singleton, meaning there is only one instance of the LOGR class across your application. This ensures consistent logging behavior but requires careful label management when shared across modules.
import { LOGR } from '@knev/bitlogr';
const LOGR1_ = LOGR.instance();
const LOGR2_ = LOGR.instance();
console.log(LOGR1_ === LOGR2_); // true (same instance)Initializing BitLOGR
To use BitLOGR, initialize the singleton instance with labels and toggle settings: 1. Labels: Define categories as an object mapping names to bit values (powers of 2). 2. Toggled: Specify which labels are active using a key-value object with boolean values.
import { LOGR, l_array } from '@knev/bitlogr';
// Define labels
const l_ = l_array(['EVENT', 'CXN', 'HANDLERS']); // { EVENT: 1, CXN: 2, HANDLERS: 4 }
// Get the logger instance
const LOGR_ = LOGR.instance();
LOGR_.labels = l_;
// Enable specific logs
LOGR_.toggled = { EVENT: true, CXN: true }; // Bitmask: 0b101 (5)Label and Toggle Formats
- Labels: An object where keys are strings and values are unique powers of 2 (e.g., 1, 2, 4, 8). These represent log categories and align with bits for efficient checking. - Example: { EVENT: 1, CXN: 2, HANDLERS: 4 }→ Bit positions 0, 1, 2.
 
- Example: 
- Toggled: An object where keys match label names and values are booleans (true/false). Internally, this creates a bitmask. - Example: { EVENT: true, CXN: false, HANDLERS: true }→ Bitmask 0b101 (5).
 
- Example: 
The toggled bitmask is compared with the log statement’s bit value using bitwise AND (&) to determine if logging occurs.
Log Statement Format
The log method takes two arguments:
- nr_logged: A number (bitmask) representing the log categories (e.g., l.EVENT | l.CXN). 
- argsFn: A thunk (function returning an array) that lazily provides log arguments. 
LOGR_.log(l_.EVENT, () => ['Debug message']); // Logs if EVENT is toggledReturn Value:
- Logging occurs only if labels are toggled true; the log function returns true/false in development (i.e., LOGR_ENABLED is true).
- undefined: in production mode, the log function returns undefined.
Using OR for Labels
Combine labels with the bitwise OR (|) operator to log under multiple categories:
LOGR_.toggled = { EVENT: true, CXN: true }; // 0b101 (5)
LOGR_.log(l_.EVENT | l_.CXN, () => ['Debug or Info']);
// Logs because 0b101 & (0b001 | 0b100) = 0b101 & 0b101 = 0b101 !== 0Utility Functions
BitLOGR provides helper functions to manage labels:
l_array(labels, start = 1)
Creates a label object from an array, assigning sequential bit values.
const l_ = l_array(['A', 'B', 'C']); // { A: 1, B: 2, C: 4 }
const l_shifted_ = l_array(['X', 'Y'], 4); // { X: 4, Y: 8 }l_length(obj)
Returns the next power of 2 based on the maximum value in the object.
const l_len_= l_length({ a: 1, b: 4 }); // 8 (next power of 2 after 4)l_concat(base, additional)
Combines label sets, appending new labels with next available bits.
const l_ = l_array(['A', 'B']); // { A: 1, B: 2 }
const l_more_ = l_concat(l_, ['C', 'D']); // { A: 1, B: 2, C: 4, D: 8 }l_merge(obj1, obj2)
Merges label sets, shifting conflicting values to unique bits.
const l1_ = { A: 1, B: 4 };
const l2_ = { C: 1, D: 8 };
const l_merged_ = l_merge(l1_, l2_); // { A: 1, B: 4, C: 16, D: 8 }l_LL(obj, shift)
Left-shifts all values in a label object.
const l_ = { A: 1, B: 2 };
const l_shifted_ = l_LL(l_, 2); // { A: 4, B: 8 }l_RR(obj, shift)
Right-shifts all values in a label object.
const l_ = { A: 4, B: 8 };
const l_shifted_ = l_RR(l_, 2); // { A: 1, B: 2 }Examples
Setting Labels
// Simple label set
const l_ = l_array(['EVENT', 'CXNS']);
LOGR_.labels = l_;Importing Labels from Submodules
import { l as l_subm_ } from './submodule.mjs'; // e.g., { EVENTS: 8, HANDLERS: 16 }
// Merge with local labels
const l_local_ = l_array(['EVENT', 'CXNS']); // { EVENT: 1, CXNS: 2 }
const l_= l_merge(l_subm_, l_local_); // { EVENTS: 8, HANDLERS: 16, EVENT: 1, CXNS: 2 }
LOGR_.labels = l_;Custom Handler
LOGR_.handler = (...args) => console.warn('WARN:', ...args);
LOGR_.toggled = { EVENT: true };
LOGR_.log(l_.EVENT, () => ['Debug warning']); // "WARN: Debug warning"Development vs. Production
Development Mode
Set LOGR_ENABLED = true (default or via build config) to enable logging:
LOGR_.toggled = { EVENT: true };
LOGR_.log(l_.EVENT, () => ['Dev log']); // Logs "Dev log"
LOGR_.log(l_.CXNS, () => ['No log']); // SkippedProduction Mode
Set LOGR_ENABLED = false(e.g., via Webpack DefinePlugin) to disable logging with no performance cost:
// In production with LOGR_ENABLED = false
LOGR_.log(l_.EVENT, () => ['Expensive computation']); // No-op, thunk not evaluatedUse a build tool to replace LOGR_ENABLED at compile time:
// webpack.config.js
const webpack = require('webpack');
module.exports = {
  plugins: [
    new webpack.DefinePlugin({
      LOGR_ENABLED: JSON.stringify(false),
    }),
  ],
};Performance Optimization with Static Functions
For critical paths, bind the log method to a static function to avoid repeated property lookups:
const log_ = LOGR_.log.bind(LOGR);
function criticalPath() {
  log_(l_.EVENT, () => ['Fast log']); // Slightly faster than logger.log
}Limitations
- 32-Bit Limit: Supports up to 31 labels before overflow (JavaScript’s 32-bit integer limit).
- Singleton Scope: Shared instance requires label consistency across modules.