mirror of
https://github.com/Mintplex-Labs/anything-llm.git
synced 2024-11-19 12:40:09 +01:00
352 lines
9.9 KiB
JavaScript
352 lines
9.9 KiB
JavaScript
const prisma = require("../utils/prisma");
|
|
const slugifyModule = require("slugify");
|
|
const { Document } = require("./documents");
|
|
const { WorkspaceUser } = require("./workspaceUsers");
|
|
const { ROLES } = require("../utils/middleware/multiUserProtected");
|
|
const { v4: uuidv4 } = require("uuid");
|
|
const { User } = require("./user");
|
|
|
|
const Workspace = {
|
|
defaultPrompt:
|
|
"Given the following conversation, relevant context, and a follow up question, reply with an answer to the current question the user is asking. Return only your response to the question given the above information following the users instructions as needed.",
|
|
writable: [
|
|
// Used for generic updates so we can validate keys in request body
|
|
"name",
|
|
"slug",
|
|
"vectorTag",
|
|
"openAiTemp",
|
|
"openAiHistory",
|
|
"lastUpdatedAt",
|
|
"openAiPrompt",
|
|
"similarityThreshold",
|
|
"chatProvider",
|
|
"chatModel",
|
|
"topN",
|
|
"chatMode",
|
|
"pfpFilename",
|
|
"agentProvider",
|
|
"agentModel",
|
|
"queryRefusalResponse",
|
|
],
|
|
/**
|
|
* The default Slugify module requires some additional mapping to prevent downstream issues
|
|
* with some vector db providers and instead of building a normalization method for every provider
|
|
* we can capture this on the table level to not have to worry about it.
|
|
* @param {...any} args - slugify args for npm package.
|
|
* @returns {string}
|
|
*/
|
|
slugify: function (...args) {
|
|
slugifyModule.extend({
|
|
"+": " plus ",
|
|
"!": " bang ",
|
|
"@": " at ",
|
|
"*": " splat ",
|
|
".": " dot ",
|
|
":": "",
|
|
"~": "",
|
|
"(": "",
|
|
")": "",
|
|
"'": "",
|
|
'"': "",
|
|
"|": "",
|
|
});
|
|
return slugifyModule(...args);
|
|
},
|
|
|
|
new: async function (name = null, creatorId = null) {
|
|
if (!name) return { result: null, message: "name cannot be null" };
|
|
var slug = this.slugify(name, { lower: true });
|
|
slug = slug || uuidv4();
|
|
|
|
const existingBySlug = await this.get({ slug });
|
|
if (existingBySlug !== null) {
|
|
const slugSeed = Math.floor(10000000 + Math.random() * 90000000);
|
|
slug = this.slugify(`${name}-${slugSeed}`, { lower: true });
|
|
}
|
|
|
|
try {
|
|
const workspace = await prisma.workspaces.create({
|
|
data: { name, slug },
|
|
});
|
|
|
|
// If created with a user then we need to create the relationship as well.
|
|
// If creating with an admin User it wont change anything because admins can
|
|
// view all workspaces anyway.
|
|
if (!!creatorId) await WorkspaceUser.create(creatorId, workspace.id);
|
|
return { workspace, message: null };
|
|
} catch (error) {
|
|
console.error(error.message);
|
|
return { workspace: null, message: error.message };
|
|
}
|
|
},
|
|
|
|
update: async function (id = null, updates = {}) {
|
|
if (!id) throw new Error("No workspace id provided for update");
|
|
|
|
const validFields = Object.keys(updates).filter((key) =>
|
|
this.writable.includes(key)
|
|
);
|
|
|
|
Object.entries(updates).forEach(([key]) => {
|
|
if (validFields.includes(key)) return;
|
|
delete updates[key];
|
|
});
|
|
|
|
if (Object.keys(updates).length === 0)
|
|
return { workspace: { id }, message: "No valid fields to update!" };
|
|
|
|
// If the user unset the chatProvider we will need
|
|
// to then clear the chatModel as well to prevent confusion during
|
|
// LLM loading.
|
|
if (updates?.chatProvider === "default") {
|
|
updates.chatProvider = null;
|
|
updates.chatModel = null;
|
|
}
|
|
|
|
return this._update(id, updates);
|
|
},
|
|
|
|
// Explicit update of settings + key validations.
|
|
// Only use this method when directly setting a key value
|
|
// that takes no user input for the keys being modified.
|
|
_update: async function (id = null, data = {}) {
|
|
if (!id) throw new Error("No workspace id provided for update");
|
|
|
|
try {
|
|
const workspace = await prisma.workspaces.update({
|
|
where: { id },
|
|
data,
|
|
});
|
|
return { workspace, message: null };
|
|
} catch (error) {
|
|
console.error(error.message);
|
|
return { workspace: null, message: error.message };
|
|
}
|
|
},
|
|
|
|
getWithUser: async function (user = null, clause = {}) {
|
|
if ([ROLES.admin, ROLES.manager].includes(user.role))
|
|
return this.get(clause);
|
|
|
|
try {
|
|
const workspace = await prisma.workspaces.findFirst({
|
|
where: {
|
|
...clause,
|
|
workspace_users: {
|
|
some: {
|
|
user_id: user?.id,
|
|
},
|
|
},
|
|
},
|
|
include: {
|
|
workspace_users: true,
|
|
documents: true,
|
|
},
|
|
});
|
|
|
|
if (!workspace) return null;
|
|
|
|
return {
|
|
...workspace,
|
|
documents: await Document.forWorkspace(workspace.id),
|
|
};
|
|
} catch (error) {
|
|
console.error(error.message);
|
|
return null;
|
|
}
|
|
},
|
|
|
|
get: async function (clause = {}) {
|
|
try {
|
|
const workspace = await prisma.workspaces.findFirst({
|
|
where: clause,
|
|
include: {
|
|
documents: true,
|
|
},
|
|
});
|
|
|
|
return workspace || null;
|
|
} catch (error) {
|
|
console.error(error.message);
|
|
return null;
|
|
}
|
|
},
|
|
|
|
delete: async function (clause = {}) {
|
|
try {
|
|
await prisma.workspaces.delete({
|
|
where: clause,
|
|
});
|
|
return true;
|
|
} catch (error) {
|
|
console.error(error.message);
|
|
return false;
|
|
}
|
|
},
|
|
|
|
where: async function (clause = {}, limit = null, orderBy = null) {
|
|
try {
|
|
const results = await prisma.workspaces.findMany({
|
|
where: clause,
|
|
...(limit !== null ? { take: limit } : {}),
|
|
...(orderBy !== null ? { orderBy } : {}),
|
|
});
|
|
return results;
|
|
} catch (error) {
|
|
console.error(error.message);
|
|
return [];
|
|
}
|
|
},
|
|
|
|
whereWithUser: async function (
|
|
user,
|
|
clause = {},
|
|
limit = null,
|
|
orderBy = null
|
|
) {
|
|
if ([ROLES.admin, ROLES.manager].includes(user.role))
|
|
return await this.where(clause, limit, orderBy);
|
|
|
|
try {
|
|
const workspaces = await prisma.workspaces.findMany({
|
|
where: {
|
|
...clause,
|
|
workspace_users: {
|
|
some: {
|
|
user_id: user.id,
|
|
},
|
|
},
|
|
},
|
|
...(limit !== null ? { take: limit } : {}),
|
|
...(orderBy !== null ? { orderBy } : {}),
|
|
});
|
|
return workspaces;
|
|
} catch (error) {
|
|
console.error(error.message);
|
|
return [];
|
|
}
|
|
},
|
|
|
|
whereWithUsers: async function (clause = {}, limit = null, orderBy = null) {
|
|
try {
|
|
const workspaces = await this.where(clause, limit, orderBy);
|
|
for (const workspace of workspaces) {
|
|
const userIds = (
|
|
await WorkspaceUser.where({ workspace_id: Number(workspace.id) })
|
|
).map((rel) => rel.user_id);
|
|
workspace.userIds = userIds;
|
|
}
|
|
return workspaces;
|
|
} catch (error) {
|
|
console.error(error.message);
|
|
return [];
|
|
}
|
|
},
|
|
|
|
workspaceUsers: async function (workspaceId) {
|
|
try {
|
|
const users = (
|
|
await WorkspaceUser.where({ workspace_id: Number(workspaceId) })
|
|
).map((rel) => rel);
|
|
|
|
const usersById = await User.where({
|
|
id: { in: users.map((user) => user.user_id) },
|
|
});
|
|
|
|
const userInfo = usersById.map((user) => {
|
|
const workspaceUser = users.find((u) => u.user_id === user.id);
|
|
return {
|
|
userId: user.id,
|
|
username: user.username,
|
|
role: user.role,
|
|
lastUpdatedAt: workspaceUser.lastUpdatedAt,
|
|
};
|
|
});
|
|
|
|
return userInfo;
|
|
} catch (error) {
|
|
console.error(error.message);
|
|
return [];
|
|
}
|
|
},
|
|
|
|
updateUsers: async function (workspaceId, userIds = []) {
|
|
try {
|
|
await WorkspaceUser.delete({ workspace_id: Number(workspaceId) });
|
|
await WorkspaceUser.createManyUsers(userIds, workspaceId);
|
|
return { success: true, error: null };
|
|
} catch (error) {
|
|
console.error(error.message);
|
|
return { success: false, error: error.message };
|
|
}
|
|
},
|
|
|
|
trackChange: async function (prevData, newData, user) {
|
|
try {
|
|
await this._trackWorkspacePromptChange(prevData, newData, user);
|
|
return;
|
|
} catch (error) {
|
|
console.error("Error tracking workspace change:", error.message);
|
|
return;
|
|
}
|
|
},
|
|
|
|
// We are only tracking this change to determine the need to a prompt library or
|
|
// prompt assistant feature. If this is something you would like to see - tell us on GitHub!
|
|
_trackWorkspacePromptChange: async function (prevData, newData, user) {
|
|
const { Telemetry } = require("./telemetry");
|
|
const { EventLogs } = require("./eventLogs");
|
|
if (
|
|
!newData?.openAiPrompt ||
|
|
newData?.openAiPrompt === this.defaultPrompt ||
|
|
newData?.openAiPrompt === prevData?.openAiPrompt
|
|
)
|
|
return;
|
|
|
|
await Telemetry.sendTelemetry("workspace_prompt_changed");
|
|
await EventLogs.logEvent(
|
|
"workspace_prompt_changed",
|
|
{
|
|
workspaceName: prevData?.name,
|
|
prevSystemPrompt: prevData?.openAiPrompt || this.defaultPrompt,
|
|
newSystemPrompt: newData?.openAiPrompt,
|
|
},
|
|
user?.id
|
|
);
|
|
return;
|
|
},
|
|
|
|
// Direct DB queries for API use only.
|
|
/**
|
|
* Generic prisma FindMany query for workspaces collections
|
|
* @param {import("../node_modules/.prisma/client/index.d.ts").Prisma.TypeMap['model']['workspaces']['operations']['findMany']['args']} prismaQuery
|
|
* @returns
|
|
*/
|
|
_findMany: async function (prismaQuery = {}) {
|
|
try {
|
|
const results = await prisma.workspaces.findMany(prismaQuery);
|
|
return results;
|
|
} catch (error) {
|
|
console.error(error.message);
|
|
return null;
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Generic prisma query for .get of workspaces collections
|
|
* @param {import("../node_modules/.prisma/client/index.d.ts").Prisma.TypeMap['model']['workspaces']['operations']['findFirst']['args']} prismaQuery
|
|
* @returns
|
|
*/
|
|
_findFirst: async function (prismaQuery = {}) {
|
|
try {
|
|
const results = await prisma.workspaces.findFirst(prismaQuery);
|
|
return results;
|
|
} catch (error) {
|
|
console.error(error.message);
|
|
return null;
|
|
}
|
|
},
|
|
};
|
|
|
|
module.exports = { Workspace };
|