489 lines
52 KiB
JavaScript
489 lines
52 KiB
JavaScript
/**
|
|
* LoRA (Low-Rank Adaptation) Runtime
|
|
*
|
|
* Efficient parameter-efficient fine-tuning adapters for LLMs.
|
|
* Supports micro-LoRA (fast, small updates) and base-LoRA (deeper adaptation).
|
|
*
|
|
* @example
|
|
* ```typescript
|
|
* import { LoraAdapter, LoraManager } from '@ruvector/ruvllm';
|
|
*
|
|
* // Create adapter
|
|
* const adapter = new LoraAdapter({
|
|
* rank: 8,
|
|
* alpha: 16,
|
|
* dropout: 0.1,
|
|
* targetModules: ['query', 'value'],
|
|
* });
|
|
*
|
|
* // Apply to hidden states
|
|
* const output = adapter.forward(hiddenStates);
|
|
*
|
|
* // Manage multiple adapters
|
|
* const manager = new LoraManager();
|
|
* manager.register('task-1', adapter);
|
|
* manager.activate('task-1');
|
|
* ```
|
|
*/
|
|
/**
|
|
* Default LoRA configuration
|
|
*/
|
|
const DEFAULT_LORA_CONFIG = {
|
|
rank: 8,
|
|
alpha: 16,
|
|
dropout: 0.1,
|
|
targetModules: ['query', 'value'],
|
|
};
|
|
/**
|
|
* LoRA Adapter
|
|
*
|
|
* Implements low-rank decomposition for parameter-efficient fine-tuning.
|
|
* W' = W + BA where A is (d x r) and B is (r x d), r << d
|
|
*
|
|
* @example
|
|
* ```typescript
|
|
* const adapter = new LoraAdapter({
|
|
* rank: 8,
|
|
* alpha: 16,
|
|
* inputDim: 768,
|
|
* outputDim: 768,
|
|
* });
|
|
*
|
|
* // Forward pass
|
|
* const output = adapter.forward(input);
|
|
*
|
|
* // Training step
|
|
* adapter.backward(input, gradOutput, 0.001);
|
|
* ```
|
|
*/
|
|
export class LoraAdapter {
|
|
constructor(config, inputDim = 256, outputDim = 256) {
|
|
this.trainingState = null;
|
|
this.frozen = false;
|
|
this.config = { ...DEFAULT_LORA_CONFIG, ...config };
|
|
this.inputDim = inputDim;
|
|
this.outputDim = outputDim;
|
|
// Initialize weights
|
|
this.weights = this.initializeWeights();
|
|
}
|
|
/**
|
|
* Forward pass through LoRA adapter
|
|
* OPTIMIZED: Uses Float64Array and loop unrolling
|
|
*
|
|
* output = input + scaling * (input @ A @ B)
|
|
*/
|
|
forward(input) {
|
|
const rank = this.config.rank;
|
|
const dim = Math.min(input.length, this.inputDim);
|
|
const scaling = this.weights.scaling;
|
|
// Apply dropout during training (simplified check)
|
|
const applyDropout = this.trainingState !== null && this.config.dropout > 0;
|
|
// input @ A (d -> r) - use typed array for hidden
|
|
const hidden = new Float64Array(rank);
|
|
for (let r = 0; r < rank; r++) {
|
|
let sum = 0;
|
|
const loraACol = this.weights.loraA;
|
|
// Unroll loop for better performance
|
|
let i = 0;
|
|
if (applyDropout) {
|
|
for (; i < dim; i++) {
|
|
if (Math.random() > this.config.dropout) {
|
|
sum += input[i] * loraACol[i][r];
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
for (; i + 3 < dim; i += 4) {
|
|
sum += input[i] * loraACol[i][r] +
|
|
input[i + 1] * loraACol[i + 1][r] +
|
|
input[i + 2] * loraACol[i + 2][r] +
|
|
input[i + 3] * loraACol[i + 3][r];
|
|
}
|
|
for (; i < dim; i++) {
|
|
sum += input[i] * loraACol[i][r];
|
|
}
|
|
}
|
|
hidden[r] = sum;
|
|
}
|
|
// hidden @ B (r -> d) + residual
|
|
const output = new Array(this.outputDim);
|
|
const loraB = this.weights.loraB;
|
|
for (let i = 0; i < this.outputDim; i++) {
|
|
let delta = 0;
|
|
for (let r = 0; r < rank; r++) {
|
|
delta += hidden[r] * loraB[r][i];
|
|
}
|
|
// Add scaled delta to input (residual connection)
|
|
output[i] = (input[i] || 0) + scaling * delta;
|
|
}
|
|
return output;
|
|
}
|
|
/**
|
|
* Forward with batch processing
|
|
*/
|
|
forwardBatch(inputs) {
|
|
return inputs.map(input => this.forward(input));
|
|
}
|
|
/**
|
|
* Backward pass and weight update
|
|
*/
|
|
backward(input, gradOutput, learningRate) {
|
|
if (this.frozen)
|
|
return 0;
|
|
const rank = this.config.rank;
|
|
const dim = Math.min(input.length, this.inputDim);
|
|
// Compute hidden activations (for gradient)
|
|
const hidden = new Array(rank).fill(0);
|
|
for (let r = 0; r < rank; r++) {
|
|
for (let i = 0; i < dim; i++) {
|
|
hidden[r] += input[i] * this.weights.loraA[i][r];
|
|
}
|
|
}
|
|
// Gradient for B: hidden^T @ gradOutput
|
|
const gradB = Array(rank).fill(null).map(() => Array(this.outputDim).fill(0));
|
|
for (let r = 0; r < rank; r++) {
|
|
for (let i = 0; i < this.outputDim; i++) {
|
|
gradB[r][i] = hidden[r] * (gradOutput[i] || 0) * this.weights.scaling;
|
|
}
|
|
}
|
|
// Gradient for hidden: gradOutput @ B^T
|
|
const gradHidden = new Array(rank).fill(0);
|
|
for (let r = 0; r < rank; r++) {
|
|
for (let i = 0; i < this.outputDim; i++) {
|
|
gradHidden[r] += (gradOutput[i] || 0) * this.weights.loraB[r][i] * this.weights.scaling;
|
|
}
|
|
}
|
|
// Gradient for A: input^T @ gradHidden
|
|
const gradA = Array(dim).fill(null).map(() => Array(rank).fill(0));
|
|
for (let i = 0; i < dim; i++) {
|
|
for (let r = 0; r < rank; r++) {
|
|
gradA[i][r] = input[i] * gradHidden[r];
|
|
}
|
|
}
|
|
// Update weights
|
|
let totalGrad = 0;
|
|
for (let i = 0; i < dim; i++) {
|
|
for (let r = 0; r < rank; r++) {
|
|
this.weights.loraA[i][r] -= learningRate * gradA[i][r];
|
|
totalGrad += Math.abs(gradA[i][r]);
|
|
}
|
|
}
|
|
for (let r = 0; r < rank; r++) {
|
|
for (let i = 0; i < this.outputDim; i++) {
|
|
this.weights.loraB[r][i] -= learningRate * gradB[r][i];
|
|
totalGrad += Math.abs(gradB[r][i]);
|
|
}
|
|
}
|
|
// Track training state
|
|
if (this.trainingState) {
|
|
this.trainingState.step++;
|
|
this.trainingState.lossHistory.push(totalGrad);
|
|
}
|
|
return totalGrad;
|
|
}
|
|
/**
|
|
* Start training mode
|
|
*/
|
|
startTraining(learningRate = 0.001) {
|
|
this.trainingState = {
|
|
step: 0,
|
|
learningRate,
|
|
gradA: Array(this.inputDim).fill(null).map(() => Array(this.config.rank).fill(0)),
|
|
gradB: Array(this.config.rank).fill(null).map(() => Array(this.outputDim).fill(0)),
|
|
lossHistory: [],
|
|
};
|
|
}
|
|
/**
|
|
* End training mode
|
|
*/
|
|
endTraining() {
|
|
const state = this.trainingState;
|
|
this.trainingState = null;
|
|
return state;
|
|
}
|
|
/**
|
|
* Freeze adapter (no more updates)
|
|
*/
|
|
freeze() {
|
|
this.frozen = true;
|
|
}
|
|
/**
|
|
* Unfreeze adapter
|
|
*/
|
|
unfreeze() {
|
|
this.frozen = false;
|
|
}
|
|
/**
|
|
* Check if frozen
|
|
*/
|
|
isFrozen() {
|
|
return this.frozen;
|
|
}
|
|
/**
|
|
* Get adapter config
|
|
*/
|
|
getConfig() {
|
|
return { ...this.config };
|
|
}
|
|
/**
|
|
* Get adapter weights
|
|
*/
|
|
getWeights() {
|
|
return {
|
|
loraA: this.weights.loraA.map(row => [...row]),
|
|
loraB: this.weights.loraB.map(row => [...row]),
|
|
scaling: this.weights.scaling,
|
|
};
|
|
}
|
|
/**
|
|
* Set adapter weights
|
|
*/
|
|
setWeights(weights) {
|
|
this.weights = {
|
|
loraA: weights.loraA.map(row => [...row]),
|
|
loraB: weights.loraB.map(row => [...row]),
|
|
scaling: weights.scaling,
|
|
};
|
|
}
|
|
/**
|
|
* Merge adapter into base weights
|
|
*
|
|
* Returns delta to add to base model weights
|
|
*/
|
|
merge() {
|
|
const delta = Array(this.inputDim)
|
|
.fill(null)
|
|
.map(() => Array(this.outputDim).fill(0));
|
|
const rank = this.config.rank;
|
|
for (let i = 0; i < this.inputDim; i++) {
|
|
for (let j = 0; j < this.outputDim; j++) {
|
|
for (let r = 0; r < rank; r++) {
|
|
delta[i][j] += this.weights.loraA[i][r] * this.weights.loraB[r][j];
|
|
}
|
|
delta[i][j] *= this.weights.scaling;
|
|
}
|
|
}
|
|
return delta;
|
|
}
|
|
/**
|
|
* Get number of trainable parameters
|
|
*/
|
|
numParameters() {
|
|
return (this.inputDim * this.config.rank) + (this.config.rank * this.outputDim);
|
|
}
|
|
/**
|
|
* Reset to initial weights
|
|
*/
|
|
reset() {
|
|
this.weights = this.initializeWeights();
|
|
this.trainingState = null;
|
|
this.frozen = false;
|
|
}
|
|
/**
|
|
* Clone adapter
|
|
*/
|
|
clone() {
|
|
const adapter = new LoraAdapter(this.config, this.inputDim, this.outputDim);
|
|
adapter.setWeights(this.getWeights());
|
|
return adapter;
|
|
}
|
|
/**
|
|
* Serialize to JSON
|
|
*/
|
|
toJSON() {
|
|
return JSON.stringify({
|
|
config: this.config,
|
|
inputDim: this.inputDim,
|
|
outputDim: this.outputDim,
|
|
weights: this.weights,
|
|
frozen: this.frozen,
|
|
});
|
|
}
|
|
/**
|
|
* Deserialize from JSON
|
|
*/
|
|
static fromJSON(json) {
|
|
const data = JSON.parse(json);
|
|
const adapter = new LoraAdapter(data.config, data.inputDim, data.outputDim);
|
|
adapter.setWeights(data.weights);
|
|
if (data.frozen)
|
|
adapter.freeze();
|
|
return adapter;
|
|
}
|
|
initializeWeights() {
|
|
const rank = this.config.rank;
|
|
// Kaiming initialization for A, zero initialization for B
|
|
const loraA = Array(this.inputDim)
|
|
.fill(null)
|
|
.map(() => Array(rank)
|
|
.fill(0)
|
|
.map(() => (Math.random() - 0.5) * Math.sqrt(2 / this.inputDim)));
|
|
const loraB = Array(rank)
|
|
.fill(null)
|
|
.map(() => Array(this.outputDim).fill(0));
|
|
return {
|
|
loraA,
|
|
loraB,
|
|
scaling: this.config.alpha / this.config.rank,
|
|
};
|
|
}
|
|
}
|
|
/**
|
|
* LoRA Manager for multiple adapters
|
|
*
|
|
* Manages a collection of LoRA adapters for different tasks/domains.
|
|
*/
|
|
export class LoraManager {
|
|
constructor(defaultConfig) {
|
|
this.adapters = new Map();
|
|
this.activeAdapterId = null;
|
|
this.defaultConfig = { ...DEFAULT_LORA_CONFIG, ...defaultConfig };
|
|
}
|
|
/**
|
|
* Register a new adapter
|
|
*/
|
|
register(id, adapter) {
|
|
this.adapters.set(id, adapter);
|
|
}
|
|
/**
|
|
* Create and register a new adapter
|
|
*/
|
|
create(id, config, inputDim, outputDim) {
|
|
const mergedConfig = { ...this.defaultConfig, ...config };
|
|
const adapter = new LoraAdapter(mergedConfig, inputDim, outputDim);
|
|
this.register(id, adapter);
|
|
return adapter;
|
|
}
|
|
/**
|
|
* Get adapter by ID
|
|
*/
|
|
get(id) {
|
|
return this.adapters.get(id);
|
|
}
|
|
/**
|
|
* Remove adapter
|
|
*/
|
|
remove(id) {
|
|
if (this.activeAdapterId === id) {
|
|
this.activeAdapterId = null;
|
|
}
|
|
return this.adapters.delete(id);
|
|
}
|
|
/**
|
|
* Activate an adapter
|
|
*/
|
|
activate(id) {
|
|
if (this.adapters.has(id)) {
|
|
this.activeAdapterId = id;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
/**
|
|
* Deactivate current adapter
|
|
*/
|
|
deactivate() {
|
|
this.activeAdapterId = null;
|
|
}
|
|
/**
|
|
* Get active adapter
|
|
*/
|
|
getActive() {
|
|
return this.activeAdapterId ? this.adapters.get(this.activeAdapterId) || null : null;
|
|
}
|
|
/**
|
|
* Get active adapter ID
|
|
*/
|
|
getActiveId() {
|
|
return this.activeAdapterId;
|
|
}
|
|
/**
|
|
* Apply active adapter
|
|
*/
|
|
forward(input) {
|
|
const active = this.getActive();
|
|
return active ? active.forward(input) : [...input];
|
|
}
|
|
/**
|
|
* List all adapter IDs
|
|
*/
|
|
list() {
|
|
return Array.from(this.adapters.keys());
|
|
}
|
|
/**
|
|
* Get adapter count
|
|
*/
|
|
count() {
|
|
return this.adapters.size;
|
|
}
|
|
/**
|
|
* Freeze all adapters
|
|
*/
|
|
freezeAll() {
|
|
for (const adapter of this.adapters.values()) {
|
|
adapter.freeze();
|
|
}
|
|
}
|
|
/**
|
|
* Unfreeze all adapters
|
|
*/
|
|
unfreezeAll() {
|
|
for (const adapter of this.adapters.values()) {
|
|
adapter.unfreeze();
|
|
}
|
|
}
|
|
/**
|
|
* Merge multiple adapters into one
|
|
*/
|
|
mergeAdapters(ids, outputId) {
|
|
const adapters = ids.map(id => this.adapters.get(id)).filter(Boolean);
|
|
if (adapters.length === 0)
|
|
return null;
|
|
// Use first adapter as base
|
|
const merged = adapters[0].clone();
|
|
const weights = merged.getWeights();
|
|
// Average weights from other adapters
|
|
for (let i = 1; i < adapters.length; i++) {
|
|
const otherWeights = adapters[i].getWeights();
|
|
for (let row = 0; row < weights.loraA.length && row < otherWeights.loraA.length; row++) {
|
|
for (let col = 0; col < weights.loraA[row].length && col < otherWeights.loraA[row].length; col++) {
|
|
weights.loraA[row][col] = (weights.loraA[row][col] + otherWeights.loraA[row][col]) / 2;
|
|
}
|
|
}
|
|
for (let row = 0; row < weights.loraB.length && row < otherWeights.loraB.length; row++) {
|
|
for (let col = 0; col < weights.loraB[row].length && col < otherWeights.loraB[row].length; col++) {
|
|
weights.loraB[row][col] = (weights.loraB[row][col] + otherWeights.loraB[row][col]) / 2;
|
|
}
|
|
}
|
|
}
|
|
merged.setWeights(weights);
|
|
this.register(outputId, merged);
|
|
return merged;
|
|
}
|
|
/**
|
|
* Get statistics
|
|
*/
|
|
stats() {
|
|
let totalParams = 0;
|
|
let frozenCount = 0;
|
|
for (const adapter of this.adapters.values()) {
|
|
totalParams += adapter.numParameters();
|
|
if (adapter.isFrozen())
|
|
frozenCount++;
|
|
}
|
|
return {
|
|
totalAdapters: this.adapters.size,
|
|
activeAdapter: this.activeAdapterId,
|
|
totalParameters: totalParams,
|
|
frozenCount,
|
|
};
|
|
}
|
|
/**
|
|
* Clear all adapters
|
|
*/
|
|
clear() {
|
|
this.adapters.clear();
|
|
this.activeAdapterId = null;
|
|
}
|
|
}
|
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lora.js","sourceRoot":"","sources":["../../src/lora.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * LoRA (Low-Rank Adaptation) Runtime\n *\n * Efficient parameter-efficient fine-tuning adapters for LLMs.\n * Supports micro-LoRA (fast, small updates) and base-LoRA (deeper adaptation).\n *\n * @example\n * ```typescript\n * import { LoraAdapter, LoraManager } from '@ruvector/ruvllm';\n *\n * // Create adapter\n * const adapter = new LoraAdapter({\n *   rank: 8,\n *   alpha: 16,\n *   dropout: 0.1,\n *   targetModules: ['query', 'value'],\n * });\n *\n * // Apply to hidden states\n * const output = adapter.forward(hiddenStates);\n *\n * // Manage multiple adapters\n * const manager = new LoraManager();\n * manager.register('task-1', adapter);\n * manager.activate('task-1');\n * ```\n */\n\nimport { LoRAConfig, Embedding } from './types';\n\n/**\n * Default LoRA configuration\n */\nconst DEFAULT_LORA_CONFIG: Required<LoRAConfig> = {\n  rank: 8,\n  alpha: 16,\n  dropout: 0.1,\n  targetModules: ['query', 'value'],\n};\n\n/**\n * LoRA adapter weights\n */\nexport interface LoraWeights {\n  /** Down projection matrix (d x r) */\n  loraA: number[][];\n  /** Up projection matrix (r x d) */\n  loraB: number[][];\n  /** Scaling factor */\n  scaling: number;\n}\n\n/**\n * LoRA training state\n */\nexport interface LoraTrainingState {\n  /** Current step */\n  step: number;\n  /** Learning rate */\n  learningRate: number;\n  /** Accumulated gradients for A */\n  gradA: number[][];\n  /** Accumulated gradients for B */\n  gradB: number[][];\n  /** Loss history */\n  lossHistory: number[];\n}\n\n/**\n * LoRA Adapter\n *\n * Implements low-rank decomposition for parameter-efficient fine-tuning.\n * W' = W + BA where A is (d x r) and B is (r x d), r << d\n *\n * @example\n * ```typescript\n * const adapter = new LoraAdapter({\n *   rank: 8,\n *   alpha: 16,\n *   inputDim: 768,\n *   outputDim: 768,\n * });\n *\n * // Forward pass\n * const output = adapter.forward(input);\n *\n * // Training step\n * adapter.backward(input, gradOutput, 0.001);\n * ```\n */\nexport class LoraAdapter {\n  private config: Required<LoRAConfig>;\n  private inputDim: number;\n  private outputDim: number;\n  private weights: LoraWeights;\n  private trainingState: LoraTrainingState | null = null;\n  private frozen: boolean = false;\n\n  constructor(config?: Partial<LoRAConfig>, inputDim = 256, outputDim = 256) {\n    this.config = { ...DEFAULT_LORA_CONFIG, ...config };\n    this.inputDim = inputDim;\n    this.outputDim = outputDim;\n\n    // Initialize weights\n    this.weights = this.initializeWeights();\n  }\n\n  /**\n   * Forward pass through LoRA adapter\n   * OPTIMIZED: Uses Float64Array and loop unrolling\n   *\n   * output = input + scaling * (input @ A @ B)\n   */\n  forward(input: number[]): number[] {\n    const rank = this.config.rank;\n    const dim = Math.min(input.length, this.inputDim);\n    const scaling = this.weights.scaling;\n\n    // Apply dropout during training (simplified check)\n    const applyDropout = this.trainingState !== null && this.config.dropout > 0;\n\n    // input @ A (d -> r) - use typed array for hidden\n    const hidden = new Float64Array(rank);\n    for (let r = 0; r < rank; r++) {\n      let sum = 0;\n      const loraACol = this.weights.loraA;\n      // Unroll loop for better performance\n      let i = 0;\n      if (applyDropout) {\n        for (; i < dim; i++) {\n          if (Math.random() > this.config.dropout) {\n            sum += input[i] * loraACol[i][r];\n          }\n        }\n      } else {\n        for (; i + 3 < dim; i += 4) {\n          sum += input[i] * loraACol[i][r] +\n                 input[i + 1] * loraACol[i + 1][r] +\n                 input[i + 2] * loraACol[i + 2][r] +\n                 input[i + 3] * loraACol[i + 3][r];\n        }\n        for (; i < dim; i++) {\n          sum += input[i] * loraACol[i][r];\n        }\n      }\n      hidden[r] = sum;\n    }\n\n    // hidden @ B (r -> d) + residual\n    const output = new Array(this.outputDim);\n    const loraB = this.weights.loraB;\n    for (let i = 0; i < this.outputDim; i++) {\n      let delta = 0;\n      for (let r = 0; r < rank; r++) {\n        delta += hidden[r] * loraB[r][i];\n      }\n      // Add scaled delta to input (residual connection)\n      output[i] = (input[i] || 0) + scaling * delta;\n    }\n\n    return output;\n  }\n\n  /**\n   * Forward with batch processing\n   */\n  forwardBatch(inputs: number[][]): number[][] {\n    return inputs.map(input => this.forward(input));\n  }\n\n  /**\n   * Backward pass and weight update\n   */\n  backward(input: number[], gradOutput: number[], learningRate: number): number {\n    if (this.frozen) return 0;\n\n    const rank = this.config.rank;\n    const dim = Math.min(input.length, this.inputDim);\n\n    // Compute hidden activations (for gradient)\n    const hidden = new Array(rank).fill(0);\n    for (let r = 0; r < rank; r++) {\n      for (let i = 0; i < dim; i++) {\n        hidden[r] += input[i] * this.weights.loraA[i][r];\n      }\n    }\n\n    // Gradient for B: hidden^T @ gradOutput\n    const gradB: number[][] = Array(rank).fill(null).map(() => Array(this.outputDim).fill(0));\n    for (let r = 0; r < rank; r++) {\n      for (let i = 0; i < this.outputDim; i++) {\n        gradB[r][i] = hidden[r] * (gradOutput[i] || 0) * this.weights.scaling;\n      }\n    }\n\n    // Gradient for hidden: gradOutput @ B^T\n    const gradHidden = new Array(rank).fill(0);\n    for (let r = 0; r < rank; r++) {\n      for (let i = 0; i < this.outputDim; i++) {\n        gradHidden[r] += (gradOutput[i] || 0) * this.weights.loraB[r][i] * this.weights.scaling;\n      }\n    }\n\n    // Gradient for A: input^T @ gradHidden\n    const gradA: number[][] = Array(dim).fill(null).map(() => Array(rank).fill(0));\n    for (let i = 0; i < dim; i++) {\n      for (let r = 0; r < rank; r++) {\n        gradA[i][r] = input[i] * gradHidden[r];\n      }\n    }\n\n    // Update weights\n    let totalGrad = 0;\n    for (let i = 0; i < dim; i++) {\n      for (let r = 0; r < rank; r++) {\n        this.weights.loraA[i][r] -= learningRate * gradA[i][r];\n        totalGrad += Math.abs(gradA[i][r]);\n      }\n    }\n    for (let r = 0; r < rank; r++) {\n      for (let i = 0; i < this.outputDim; i++) {\n        this.weights.loraB[r][i] -= learningRate * gradB[r][i];\n        totalGrad += Math.abs(gradB[r][i]);\n      }\n    }\n\n    // Track training state\n    if (this.trainingState) {\n      this.trainingState.step++;\n      this.trainingState.lossHistory.push(totalGrad);\n    }\n\n    return totalGrad;\n  }\n\n  /**\n   * Start training mode\n   */\n  startTraining(learningRate = 0.001): void {\n    this.trainingState = {\n      step: 0,\n      learningRate,\n      gradA: Array(this.inputDim).fill(null).map(() => Array(this.config.rank).fill(0)),\n      gradB: Array(this.config.rank).fill(null).map(() => Array(this.outputDim).fill(0)),\n      lossHistory: [],\n    };\n  }\n\n  /**\n   * End training mode\n   */\n  endTraining(): LoraTrainingState | null {\n    const state = this.trainingState;\n    this.trainingState = null;\n    return state;\n  }\n\n  /**\n   * Freeze adapter (no more updates)\n   */\n  freeze(): void {\n    this.frozen = true;\n  }\n\n  /**\n   * Unfreeze adapter\n   */\n  unfreeze(): void {\n    this.frozen = false;\n  }\n\n  /**\n   * Check if frozen\n   */\n  isFrozen(): boolean {\n    return this.frozen;\n  }\n\n  /**\n   * Get adapter config\n   */\n  getConfig(): Required<LoRAConfig> {\n    return { ...this.config };\n  }\n\n  /**\n   * Get adapter weights\n   */\n  getWeights(): LoraWeights {\n    return {\n      loraA: this.weights.loraA.map(row => [...row]),\n      loraB: this.weights.loraB.map(row => [...row]),\n      scaling: this.weights.scaling,\n    };\n  }\n\n  /**\n   * Set adapter weights\n   */\n  setWeights(weights: LoraWeights): void {\n    this.weights = {\n      loraA: weights.loraA.map(row => [...row]),\n      loraB: weights.loraB.map(row => [...row]),\n      scaling: weights.scaling,\n    };\n  }\n\n  /**\n   * Merge adapter into base weights\n   *\n   * Returns delta to add to base model weights\n   */\n  merge(): number[][] {\n    const delta: number[][] = Array(this.inputDim)\n      .fill(null)\n      .map(() => Array(this.outputDim).fill(0));\n\n    const rank = this.config.rank;\n    for (let i = 0; i < this.inputDim; i++) {\n      for (let j = 0; j < this.outputDim; j++) {\n        for (let r = 0; r < rank; r++) {\n          delta[i][j] += this.weights.loraA[i][r] * this.weights.loraB[r][j];\n        }\n        delta[i][j] *= this.weights.scaling;\n      }\n    }\n\n    return delta;\n  }\n\n  /**\n   * Get number of trainable parameters\n   */\n  numParameters(): number {\n    return (this.inputDim * this.config.rank) + (this.config.rank * this.outputDim);\n  }\n\n  /**\n   * Reset to initial weights\n   */\n  reset(): void {\n    this.weights = this.initializeWeights();\n    this.trainingState = null;\n    this.frozen = false;\n  }\n\n  /**\n   * Clone adapter\n   */\n  clone(): LoraAdapter {\n    const adapter = new LoraAdapter(this.config, this.inputDim, this.outputDim);\n    adapter.setWeights(this.getWeights());\n    return adapter;\n  }\n\n  /**\n   * Serialize to JSON\n   */\n  toJSON(): string {\n    return JSON.stringify({\n      config: this.config,\n      inputDim: this.inputDim,\n      outputDim: this.outputDim,\n      weights: this.weights,\n      frozen: this.frozen,\n    });\n  }\n\n  /**\n   * Deserialize from JSON\n   */\n  static fromJSON(json: string): LoraAdapter {\n    const data = JSON.parse(json);\n    const adapter = new LoraAdapter(data.config, data.inputDim, data.outputDim);\n    adapter.setWeights(data.weights);\n    if (data.frozen) adapter.freeze();\n    return adapter;\n  }\n\n  private initializeWeights(): LoraWeights {\n    const rank = this.config.rank;\n\n    // Kaiming initialization for A, zero initialization for B\n    const loraA: number[][] = Array(this.inputDim)\n      .fill(null)\n      .map(() =>\n        Array(rank)\n          .fill(0)\n          .map(() => (Math.random() - 0.5) * Math.sqrt(2 / this.inputDim))\n      );\n\n    const loraB: number[][] = Array(rank)\n      .fill(null)\n      .map(() => Array(this.outputDim).fill(0));\n\n    return {\n      loraA,\n      loraB,\n      scaling: this.config.alpha / this.config.rank,\n    };\n  }\n}\n\n/**\n * LoRA Manager for multiple adapters\n *\n * Manages a collection of LoRA adapters for different tasks/domains.\n */\nexport class LoraManager {\n  private adapters: Map<string, LoraAdapter> = new Map();\n  private activeAdapterId: string | null = null;\n  private defaultConfig: Required<LoRAConfig>;\n\n  constructor(defaultConfig?: Partial<LoRAConfig>) {\n    this.defaultConfig = { ...DEFAULT_LORA_CONFIG, ...defaultConfig };\n  }\n\n  /**\n   * Register a new adapter\n   */\n  register(id: string, adapter: LoraAdapter): void {\n    this.adapters.set(id, adapter);\n  }\n\n  /**\n   * Create and register a new adapter\n   */\n  create(id: string, config?: Partial<LoRAConfig>, inputDim?: number, outputDim?: number): LoraAdapter {\n    const mergedConfig = { ...this.defaultConfig, ...config };\n    const adapter = new LoraAdapter(mergedConfig, inputDim, outputDim);\n    this.register(id, adapter);\n    return adapter;\n  }\n\n  /**\n   * Get adapter by ID\n   */\n  get(id: string): LoraAdapter | undefined {\n    return this.adapters.get(id);\n  }\n\n  /**\n   * Remove adapter\n   */\n  remove(id: string): boolean {\n    if (this.activeAdapterId === id) {\n      this.activeAdapterId = null;\n    }\n    return this.adapters.delete(id);\n  }\n\n  /**\n   * Activate an adapter\n   */\n  activate(id: string): boolean {\n    if (this.adapters.has(id)) {\n      this.activeAdapterId = id;\n      return true;\n    }\n    return false;\n  }\n\n  /**\n   * Deactivate current adapter\n   */\n  deactivate(): void {\n    this.activeAdapterId = null;\n  }\n\n  /**\n   * Get active adapter\n   */\n  getActive(): LoraAdapter | null {\n    return this.activeAdapterId ? this.adapters.get(this.activeAdapterId) || null : null;\n  }\n\n  /**\n   * Get active adapter ID\n   */\n  getActiveId(): string | null {\n    return this.activeAdapterId;\n  }\n\n  /**\n   * Apply active adapter\n   */\n  forward(input: number[]): number[] {\n    const active = this.getActive();\n    return active ? active.forward(input) : [...input];\n  }\n\n  /**\n   * List all adapter IDs\n   */\n  list(): string[] {\n    return Array.from(this.adapters.keys());\n  }\n\n  /**\n   * Get adapter count\n   */\n  count(): number {\n    return this.adapters.size;\n  }\n\n  /**\n   * Freeze all adapters\n   */\n  freezeAll(): void {\n    for (const adapter of this.adapters.values()) {\n      adapter.freeze();\n    }\n  }\n\n  /**\n   * Unfreeze all adapters\n   */\n  unfreezeAll(): void {\n    for (const adapter of this.adapters.values()) {\n      adapter.unfreeze();\n    }\n  }\n\n  /**\n   * Merge multiple adapters into one\n   */\n  mergeAdapters(ids: string[], outputId: string): LoraAdapter | null {\n    const adapters = ids.map(id => this.adapters.get(id)).filter(Boolean) as LoraAdapter[];\n    if (adapters.length === 0) return null;\n\n    // Use first adapter as base\n    const merged = adapters[0].clone();\n    const weights = merged.getWeights();\n\n    // Average weights from other adapters\n    for (let i = 1; i < adapters.length; i++) {\n      const otherWeights = adapters[i].getWeights();\n\n      for (let row = 0; row < weights.loraA.length && row < otherWeights.loraA.length; row++) {\n        for (let col = 0; col < weights.loraA[row].length && col < otherWeights.loraA[row].length; col++) {\n          weights.loraA[row][col] = (weights.loraA[row][col] + otherWeights.loraA[row][col]) / 2;\n        }\n      }\n      for (let row = 0; row < weights.loraB.length && row < otherWeights.loraB.length; row++) {\n        for (let col = 0; col < weights.loraB[row].length && col < otherWeights.loraB[row].length; col++) {\n          weights.loraB[row][col] = (weights.loraB[row][col] + otherWeights.loraB[row][col]) / 2;\n        }\n      }\n    }\n\n    merged.setWeights(weights);\n    this.register(outputId, merged);\n    return merged;\n  }\n\n  /**\n   * Get statistics\n   */\n  stats(): {\n    totalAdapters: number;\n    activeAdapter: string | null;\n    totalParameters: number;\n    frozenCount: number;\n  } {\n    let totalParams = 0;\n    let frozenCount = 0;\n\n    for (const adapter of this.adapters.values()) {\n      totalParams += adapter.numParameters();\n      if (adapter.isFrozen()) frozenCount++;\n    }\n\n    return {\n      totalAdapters: this.adapters.size,\n      activeAdapter: this.activeAdapterId,\n      totalParameters: totalParams,\n      frozenCount,\n    };\n  }\n\n  /**\n   * Clear all adapters\n   */\n  clear(): void {\n    this.adapters.clear();\n    this.activeAdapterId = null;\n  }\n}\n"]}
|