mirror of
https://github.com/danny-avila/LibreChat.git
synced 2025-09-22 06:00:56 +02:00

* 🚀 feat: Introduce data schemas and refactor models to use @librechat/data-schemas * 🚀 feat: Add installation step for Data Schemas Package in backend review workflow * chore: Add `data-schemas` package to update/rebuild packages scripts * chore: Update Dockerfile to include data-schemas package build process * fix: add missing @rollup/plugin-typescript package * chore: Add GitHub Actions workflow for publishing data-schemas package --------- Co-authored-by: Danny Avila <danny@librechat.ai>
187 lines
5.8 KiB
JavaScript
187 lines
5.8 KiB
JavaScript
const mongoose = require('mongoose');
|
|
const { isEnabled } = require('~/server/utils/handleText');
|
|
const { transactionSchema } = require('@librechat/data-schemas');
|
|
const { getMultiplier, getCacheMultiplier } = require('./tx');
|
|
const { logger } = require('~/config');
|
|
const Balance = require('./Balance');
|
|
const cancelRate = 1.15;
|
|
|
|
/** Method to calculate and set the tokenValue for a transaction */
|
|
transactionSchema.methods.calculateTokenValue = function () {
|
|
if (!this.valueKey || !this.tokenType) {
|
|
this.tokenValue = this.rawAmount;
|
|
}
|
|
const { valueKey, tokenType, model, endpointTokenConfig } = this;
|
|
const multiplier = Math.abs(getMultiplier({ valueKey, tokenType, model, endpointTokenConfig }));
|
|
this.rate = multiplier;
|
|
this.tokenValue = this.rawAmount * multiplier;
|
|
if (this.context && this.tokenType === 'completion' && this.context === 'incomplete') {
|
|
this.tokenValue = Math.ceil(this.tokenValue * cancelRate);
|
|
this.rate *= cancelRate;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Static method to create a transaction and update the balance
|
|
* @param {txData} txData - Transaction data.
|
|
*/
|
|
transactionSchema.statics.create = async function (txData) {
|
|
const Transaction = this;
|
|
if (txData.rawAmount != null && isNaN(txData.rawAmount)) {
|
|
return;
|
|
}
|
|
|
|
const transaction = new Transaction(txData);
|
|
transaction.endpointTokenConfig = txData.endpointTokenConfig;
|
|
transaction.calculateTokenValue();
|
|
|
|
await transaction.save();
|
|
|
|
if (!isEnabled(process.env.CHECK_BALANCE)) {
|
|
return;
|
|
}
|
|
|
|
let balance = await Balance.findOne({ user: transaction.user }).lean();
|
|
let incrementValue = transaction.tokenValue;
|
|
|
|
if (balance && balance?.tokenCredits + incrementValue < 0) {
|
|
incrementValue = -balance.tokenCredits;
|
|
}
|
|
|
|
balance = await Balance.findOneAndUpdate(
|
|
{ user: transaction.user },
|
|
{ $inc: { tokenCredits: incrementValue } },
|
|
{ upsert: true, new: true },
|
|
).lean();
|
|
|
|
return {
|
|
rate: transaction.rate,
|
|
user: transaction.user.toString(),
|
|
balance: balance.tokenCredits,
|
|
[transaction.tokenType]: incrementValue,
|
|
};
|
|
};
|
|
|
|
/**
|
|
* Static method to create a structured transaction and update the balance
|
|
* @param {txData} txData - Transaction data.
|
|
*/
|
|
transactionSchema.statics.createStructured = async function (txData) {
|
|
const Transaction = this;
|
|
|
|
const transaction = new Transaction({
|
|
...txData,
|
|
endpointTokenConfig: txData.endpointTokenConfig,
|
|
});
|
|
|
|
transaction.calculateStructuredTokenValue();
|
|
|
|
await transaction.save();
|
|
|
|
if (!isEnabled(process.env.CHECK_BALANCE)) {
|
|
return;
|
|
}
|
|
|
|
let balance = await Balance.findOne({ user: transaction.user }).lean();
|
|
let incrementValue = transaction.tokenValue;
|
|
|
|
if (balance && balance?.tokenCredits + incrementValue < 0) {
|
|
incrementValue = -balance.tokenCredits;
|
|
}
|
|
|
|
balance = await Balance.findOneAndUpdate(
|
|
{ user: transaction.user },
|
|
{ $inc: { tokenCredits: incrementValue } },
|
|
{ upsert: true, new: true },
|
|
).lean();
|
|
|
|
return {
|
|
rate: transaction.rate,
|
|
user: transaction.user.toString(),
|
|
balance: balance.tokenCredits,
|
|
[transaction.tokenType]: incrementValue,
|
|
};
|
|
};
|
|
|
|
/** Method to calculate token value for structured tokens */
|
|
transactionSchema.methods.calculateStructuredTokenValue = function () {
|
|
if (!this.tokenType) {
|
|
this.tokenValue = this.rawAmount;
|
|
return;
|
|
}
|
|
|
|
const { model, endpointTokenConfig } = this;
|
|
|
|
if (this.tokenType === 'prompt') {
|
|
const inputMultiplier = getMultiplier({ tokenType: 'prompt', model, endpointTokenConfig });
|
|
const writeMultiplier =
|
|
getCacheMultiplier({ cacheType: 'write', model, endpointTokenConfig }) ?? inputMultiplier;
|
|
const readMultiplier =
|
|
getCacheMultiplier({ cacheType: 'read', model, endpointTokenConfig }) ?? inputMultiplier;
|
|
|
|
this.rateDetail = {
|
|
input: inputMultiplier,
|
|
write: writeMultiplier,
|
|
read: readMultiplier,
|
|
};
|
|
|
|
const totalPromptTokens =
|
|
Math.abs(this.inputTokens || 0) +
|
|
Math.abs(this.writeTokens || 0) +
|
|
Math.abs(this.readTokens || 0);
|
|
|
|
if (totalPromptTokens > 0) {
|
|
this.rate =
|
|
(Math.abs(inputMultiplier * (this.inputTokens || 0)) +
|
|
Math.abs(writeMultiplier * (this.writeTokens || 0)) +
|
|
Math.abs(readMultiplier * (this.readTokens || 0))) /
|
|
totalPromptTokens;
|
|
} else {
|
|
this.rate = Math.abs(inputMultiplier); // Default to input rate if no tokens
|
|
}
|
|
|
|
this.tokenValue = -(
|
|
Math.abs(this.inputTokens || 0) * inputMultiplier +
|
|
Math.abs(this.writeTokens || 0) * writeMultiplier +
|
|
Math.abs(this.readTokens || 0) * readMultiplier
|
|
);
|
|
|
|
this.rawAmount = -totalPromptTokens;
|
|
} else if (this.tokenType === 'completion') {
|
|
const multiplier = getMultiplier({ tokenType: this.tokenType, model, endpointTokenConfig });
|
|
this.rate = Math.abs(multiplier);
|
|
this.tokenValue = -Math.abs(this.rawAmount) * multiplier;
|
|
this.rawAmount = -Math.abs(this.rawAmount);
|
|
}
|
|
|
|
if (this.context && this.tokenType === 'completion' && this.context === 'incomplete') {
|
|
this.tokenValue = Math.ceil(this.tokenValue * cancelRate);
|
|
this.rate *= cancelRate;
|
|
if (this.rateDetail) {
|
|
this.rateDetail = Object.fromEntries(
|
|
Object.entries(this.rateDetail).map(([k, v]) => [k, v * cancelRate]),
|
|
);
|
|
}
|
|
}
|
|
};
|
|
|
|
const Transaction = mongoose.model('Transaction', transactionSchema);
|
|
|
|
/**
|
|
* Queries and retrieves transactions based on a given filter.
|
|
* @async
|
|
* @function getTransactions
|
|
* @param {Object} filter - MongoDB filter object to apply when querying transactions.
|
|
* @returns {Promise<Array>} A promise that resolves to an array of matched transactions.
|
|
* @throws {Error} Throws an error if querying the database fails.
|
|
*/
|
|
async function getTransactions(filter) {
|
|
try {
|
|
return await Transaction.find(filter).lean();
|
|
} catch (error) {
|
|
logger.error('Error querying transactions:', error);
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
module.exports = { Transaction, getTransactions };
|