core/apps/webapp/app/models/workspace.server.ts
Harshith Mullapudi bcc0560cf0
Feat: Space (#93)
* Feat: change space assignment from statement to episode

* feat: add default spaces and improve integration, space tools discovery in MCP

* feat: change spaces to episode based

* Feat: take multiple spaceIds while ingesting

* Feat: modify mcp tool descriptions, add spaceId in mcp url

* feat: add copy

* bump: new version 0.1.24

---------

Co-authored-by: Manoj <saimanoj58@gmail.com>
2025-10-09 12:38:42 +05:30

103 lines
3.5 KiB
TypeScript

import { type Workspace } from "@core/database";
import { prisma } from "~/db.server";
import { ensureBillingInitialized } from "~/services/billing.server";
import { sendEmail } from "~/services/email.server";
import { logger } from "~/services/logger.service";
import { SpaceService } from "~/services/space.server";
interface CreateWorkspaceDto {
name: string;
integrations: string[];
userId: string;
}
const spaceService = new SpaceService();
const profileRule = `
Purpose: Store my identity and preferences to improve personalization across assistants. It should be broadly useful across contexts (not app-specific).
Include (examples):
• Preferred name, pronunciation, public handles (GitHub/Twitter/LinkedIn URLs), primary email domain
• Timezone, locale, working hours, meeting preferences (async/sync bias, default duration)
• Role, team, company, office location (city-level only), seniority
• Tooling defaults (editor, ticketing system, repo host), keyboard layout, OS
• Communication preferences (tone, brevity vs. detail, summary-first)
Exclude:
• Sensitive: secrets, health/financial/political/religious/sexual data, precise address
• Temporary: one-off states, troubleshooting sessions, query results
• Context-specific: app behaviors, work conversations, project-specific preferences
• Meta: discussions about this memory system, AI architecture, system design
• Anything not explicitly consented to share
don't store anything the user did not explicitly consent to share.`;
const githubDescription = `Everything related to my GitHub work - repos I'm working on, projects I contribute to, code I'm writing, PRs I'm reviewing. Basically my coding life on GitHub.`;
const healthDescription = `My health and wellness stuff - how I'm feeling, what I'm learning about my body, experiments I'm trying, patterns I notice. Whatever matters to me about staying healthy.`;
const fitnessDescription = `My workouts and training - what I'm doing at the gym, runs I'm going on, progress I'm making, goals I'm chasing. Anything related to physical exercise and getting stronger.`;
export async function createWorkspace(
input: CreateWorkspaceDto,
): Promise<Workspace> {
const workspace = await prisma.workspace.create({
data: {
slug: input.name,
name: input.name,
userId: input.userId,
},
});
const user = await prisma.user.update({
where: { id: input.userId },
data: {
confirmedBasicDetails: true,
},
});
await ensureBillingInitialized(workspace.id);
// Create default spaces
await Promise.all([
spaceService.createSpace({
name: "Profile",
description: profileRule,
userId: input.userId,
workspaceId: workspace.id,
}),
spaceService.createSpace({
name: "GitHub",
description: githubDescription,
userId: input.userId,
workspaceId: workspace.id,
}),
spaceService.createSpace({
name: "Health",
description: healthDescription,
userId: input.userId,
workspaceId: workspace.id,
}),
spaceService.createSpace({
name: "Fitness",
description: fitnessDescription,
userId: input.userId,
workspaceId: workspace.id,
}),
]);
try {
const response = await sendEmail({ email: "welcome", to: user.email });
logger.info(`${JSON.stringify(response)}`);
} catch (e) {
logger.error(`Error sending email: ${e}`);
}
return workspace;
}
export async function getWorkspaceByUser(userId: string) {
return await prisma.workspace.findFirst({
where: {
userId,
},
});
}