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.
- 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.
Install via npm:
npm install @knev/bitlogr
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)
To use BitLOGR, initialize the singleton instance with labels and toggle settings:
- Labels: Define categories as an object mapping names to bit values (powers of 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)
-
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.
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 toggled
Return 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.
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 !== 0
BitLOGR provides helper functions to manage labels:
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 }
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)
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 }
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 }
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 }
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 }
// Simple label set
const l_ = l_array(['EVENT', 'CXNS']);
LOGR_.labels = l_;
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_;
LOGR_.handler = (...args) => console.warn('WARN:', ...args);
LOGR_.toggled = { EVENT: true };
LOGR_.log(l_.EVENT, () => ['Debug warning']); // "WARN: Debug warning"
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']); // Skipped
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 evaluated
Use 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),
}),
],
};
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
}
- 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.