@danielcok17/prisma-db 1.4.0 → 1.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,5 +1,5 @@
1
1
  {
2
- "name": "prisma-client-f94f7b73f322567a3aaf389dd6cbee3e8c3293b739d8157fa3935881580615f1",
2
+ "name": "prisma-client-db68099bf6090dcfebda77a1130bfd13cc86c226f2e9eb2f7fbc7b2195f44cb6",
3
3
  "main": "index.js",
4
4
  "types": "index.d.ts",
5
5
  "browser": "default.js",
@@ -516,10 +516,11 @@ enum SubscriptionStatus {
516
516
 
517
517
  // Stripe: Tier predplatného
518
518
  enum SubscriptionTier {
519
- FREE // Bezplatný tier (10 správ)
520
- LAWYER // Právnik tier (1000 správ, €49/mes, 1 user)
521
- LAW_FIRM // Právna kancelária tier (5000 správ, €149/mes, 5 users)
522
- ENTERPRISE // Enterprise tier (unlimited správ, unlimited users, custom price)
519
+ FREE // Free tier (10 messages/month)
520
+ LAWYER // Lawyer tier (1000 messages/month, €29/month, 1 user)
521
+ LAWYER_PRO // Lawyer Pro tier (3000 messages/month, €49/month, 1 user) - RECOMMENDED
522
+ LAW_FIRM // Law Firm tier (10000 messages/month, €129/month, up to 5 users)
523
+ ENTERPRISE // Enterprise tier (unlimited messages, unlimited users, custom pricing)
523
524
  }
524
525
 
525
526
  // Stripe: Billing interval
@@ -480,6 +480,7 @@ exports.SubscriptionStatus = exports.$Enums.SubscriptionStatus = {
480
480
  exports.SubscriptionTier = exports.$Enums.SubscriptionTier = {
481
481
  FREE: 'FREE',
482
482
  LAWYER: 'LAWYER',
483
+ LAWYER_PRO: 'LAWYER_PRO',
483
484
  LAW_FIRM: 'LAW_FIRM',
484
485
  ENTERPRISE: 'ENTERPRISE'
485
486
  };
@@ -579,7 +580,6 @@ const config = {
579
580
  "db"
580
581
  ],
581
582
  "activeProvider": "postgresql",
582
- "postinstall": false,
583
583
  "inlineDatasources": {
584
584
  "db": {
585
585
  "url": {
@@ -588,8 +588,8 @@ const config = {
588
588
  }
589
589
  }
590
590
  },
591
- "inlineSchema": "generator client {\n provider = \"prisma-client-js\"\n output = \"./generated/app\"\n binaryTargets = [\"native\", \"linux-musl-openssl-3.0.x\"]\n}\n\ndatasource db {\n provider = \"postgresql\"\n url = env(\"POSTGRES_PRISMA_URL\")\n directUrl = env(\"POSTGRES_URL_NON_POOLING\")\n}\n\nmodel Account {\n id String @id @default(cuid())\n userId String\n type String\n provider String\n providerAccountId String\n refresh_token String?\n access_token String?\n expires_at Int?\n token_type String?\n scope String?\n id_token String?\n session_state String?\n user User @relation(fields: [userId], references: [id], onDelete: Cascade)\n\n @@unique([provider, providerAccountId])\n}\n\nmodel User {\n id String @id @default(cuid())\n name String?\n email String? @unique\n emailVerified DateTime?\n image String?\n // Credentials authentication fields\n password String? // Hashed password pre credentials login\n createdAt DateTime @default(now())\n agreedToTerms Boolean @default(false)\n practiceArea String[]\n lawFirm String?\n yearsOfExperience Int?\n // Nové polia pre schvalovanie používateľov\n isApproved Boolean @default(false) // Či je používateľ schválený\n isRejected Boolean @default(false) // Či je používateľ zamietnutý\n approvedAt DateTime? // Kedy bol schválený\n rejectedAt DateTime? // Kedy bol zamietnutý\n approvedBy String? // ID admina, ktorý schválil\n rejectedBy String? // ID admina, ktorý zamietol\n rejectionReason String? // Dôvod zamietnutia\n // Nové polia pre tracking a žiadosť\n referralSource String? // Odkiaľ sa o nás dozvedel (Google, Facebook, LinkedIn, etc.)\n applicationText String? // Text žiadosti o prihlásenie\n // ✨ STRIPE FIELDS (B2C - len pre individuálnych používateľov)\n subscriptionTier SubscriptionTier @default(FREE) // Aktuálny subscription tier (cache)\n messageCount Int @default(10) // ✅ OPRAVENÉ z 100 na 10\n messageCountResetAt DateTime? // Kedy sa má resetovať message count\n // Relations\n approvalRequest UserApprovalRequest?\n stripeCustomer StripeCustomer? // ✨ B2C Stripe customer\n ownedOrganizations Organization[] @relation(\"OrganizationOwner\")\n organizationMembers OrganizationMember[]\n accounts Account[]\n answers Answer[]\n conversations Conversation[]\n feedbacks Feedback[]\n pageViews PageView[]\n sessions Session[]\n workflowLogs WorkflowLog[]\n verificationTokens VerificationToken[]\n passwordResetTokens PasswordResetToken[]\n}\n\n// Nový model pre žiadosti o schválenie\nmodel UserApprovalRequest {\n id String @id @default(cuid())\n userId String @unique\n status ApprovalStatus @default(PENDING)\n submittedAt DateTime @default(now())\n reviewedAt DateTime?\n reviewedBy String? // ID admina, ktorý preskúmal žiadosť\n notes String? // Poznámky admina\n user User @relation(fields: [userId], references: [id], onDelete: Cascade)\n\n @@index([status])\n @@index([submittedAt])\n @@index([reviewedAt])\n}\n\n// ============================================\n// ORGANIZATION MODELS (B2B)\n// ============================================\n\n// Organization = Právna kancelária / Firma\nmodel Organization {\n id String @id @default(cuid())\n name String // \"Advokátska kancelária XYZ s.r.o.\"\n\n // Billing & Legal Information\n companyNumber String? // IČO (Identifikačné číslo organizácie)\n vatNumber String? // IČ DPH (VAT Number)\n taxNumber String? // DIČ (Daňové identifikačné číslo)\n legalForm String? // \"s.r.o.\", \"a.s.\", \"v.o.s.\", \"k.s.\", \"SZČO\"\n billingEmail String\n\n // Address\n street String?\n city String?\n postalCode String?\n country String @default(\"SK\")\n\n // Subscription (B2B)\n subscriptionTier SubscriptionTier @default(FREE) // Tier organizácie (cache)\n messageCount Int @default(0) // Spoločný pool správ pre všetkých členov\n messageLimit Int @default(100) // Limit správ podľa tieru\n messageCountResetAt DateTime? // Kedy sa má resetovať pool\n\n // Settings\n isActive Boolean @default(true)\n maxMembers Int @default(5) // Max počet členov podľa tieru\n\n // Relations\n owner User @relation(\"OrganizationOwner\", fields: [ownerId], references: [id])\n ownerId String\n members OrganizationMember[]\n stripeCustomer StripeCustomer?\n\n createdAt DateTime @default(now())\n updatedAt DateTime @updatedAt\n\n @@index([ownerId])\n @@index([vatNumber])\n @@index([companyNumber])\n @@index([subscriptionTier])\n @@index([billingEmail])\n}\n\n// Many-to-Many: User ↔ Organization\nmodel OrganizationMember {\n id String @id @default(cuid())\n organizationId String\n userId String\n role MemberRole @default(MEMBER)\n joinedAt DateTime @default(now())\n\n organization Organization @relation(fields: [organizationId], references: [id], onDelete: Cascade)\n user User @relation(fields: [userId], references: [id], onDelete: Cascade)\n\n @@unique([organizationId, userId])\n @@index([organizationId])\n @@index([userId])\n @@index([role])\n}\n\nenum MemberRole {\n OWNER // Zakladateľ/vlastník, plný prístup k billing a settings\n ADMIN // Administrátor, môže pridávať/odoberať členov\n MEMBER // Bežný člen, len používa systém\n}\n\n// ZJEDNODUŠENÝ Conversation - len metadata, bez duplikátov\nmodel Conversation {\n id String @id @default(cuid())\n name String\n userId String\n isShareable Boolean @default(false)\n shareUrl String? @unique\n sharedAt DateTime?\n createdAt DateTime @default(now())\n updatedAt DateTime @updatedAt\n summary String?\n messagesSinceLastSummary Int? @default(0)\n // Relácie\n answers Answer[]\n user User @relation(fields: [userId], references: [id])\n workflowLogs WorkflowLog[]\n\n @@index([userId])\n @@index([createdAt])\n}\n\n// Hlavný model - všetky správy, bez duplikátov\nmodel Answer {\n id String @id @default(cuid())\n conversationId String\n messageId String @unique\n role Role\n content String\n question String?\n answer String?\n evaluation String?\n isWelcome Boolean @default(false)\n processingTime Int?\n model String?\n userId String?\n createdAt DateTime @default(now())\n updatedAt DateTime @updatedAt\n // Relácie\n conversation Conversation @relation(fields: [conversationId], references: [id], onDelete: Cascade)\n user User? @relation(fields: [userId], references: [id])\n feedback Feedback?\n references Reference[]\n metrics AnswerMetrics?\n WorkflowLog WorkflowLog[]\n files MessageFile[]\n\n @@index([conversationId])\n @@index([messageId])\n @@index([role])\n @@index([userId])\n @@index([createdAt])\n}\n\nmodel MessageFile {\n id String @id @default(cuid())\n answerId String\n fileName String\n fileType String\n base64Data String\n uploadedAt DateTime @default(now())\n answer Answer @relation(fields: [answerId], references: [id], onDelete: Cascade)\n\n @@index([answerId])\n @@index([fileType])\n @@index([uploadedAt])\n}\n\n// Nová tabuľka pre metriky odpovedí\nmodel AnswerMetrics {\n id String @id @default(cuid())\n answerId String @unique\n // Náklady\n apiCost Float? // Celkové náklady (ai_cost + rag_cost)\n aiCost Float? // Náklady len na AI provider volania\n ragCost Float? // Náklady na RAG operácie\n // Volania\n apiCalls Int? // Celkový počet volaní\n aiCalls Int? // Počet AI provider volaní\n ragCalls Int? // Počet RAG operácií\n // Čas a výkon\n apiDuration Int? // Doba spracovania v ms\n // AI Provider a Model\n aiProvider String? // Hlavný AI provider\n aiModel String? // Hlavný AI model\n aiProvidersUsed String[] // Všetky použité AI providery\n aiModelsUsed String[] // Všetky použité AI modely\n // Timestamps\n createdAt DateTime @default(now())\n updatedAt DateTime @updatedAt\n // Relations\n answer Answer @relation(fields: [answerId], references: [id], onDelete: Cascade)\n\n @@index([answerId])\n @@index([apiCost])\n @@index([apiDuration])\n @@index([aiProvider])\n @@index([createdAt])\n}\n\nmodel Feedback {\n id String @id @default(cuid())\n answerId String @unique\n rating FeedbackRating\n feedbackTypes String[]\n customFeedback String?\n userId String\n messageContent String?\n createdAt DateTime @default(now())\n answer Answer @relation(fields: [answerId], references: [id], onDelete: Cascade)\n user User @relation(fields: [userId], references: [id])\n\n @@index([rating])\n @@index([userId])\n @@index([answerId])\n}\n\n// ZJEDNODUŠENÝ Reference - len na Answer, bez duplikátov\nmodel Reference {\n id String @id @default(cuid())\n answerId String // Len answerId - konzistentné\n type ReferenceType\n title String\n reference String\n summary String?\n url String?\n uuid String?\n relevance Float?\n citationNumber Int?\n metadata Json?\n userId String?\n createdAt DateTime @default(now())\n // Relácie\n answer Answer @relation(fields: [answerId], references: [id], onDelete: Cascade)\n\n @@index([answerId])\n @@index([type])\n @@index([reference])\n @@index([userId])\n @@index([createdAt])\n}\n\nmodel PageView {\n id String @id @default(cuid())\n userId String?\n sessionId String?\n page String\n path String\n referrer String?\n userAgent String?\n ipAddress String?\n country String?\n city String?\n deviceType String?\n browser String?\n os String?\n screenSize String?\n language String?\n timeOnPage Int?\n isBounce Boolean @default(true)\n createdAt DateTime @default(now())\n session Session? @relation(fields: [sessionId], references: [sessionId], onDelete: SetNull)\n user User? @relation(fields: [userId], references: [id])\n\n @@index([userId])\n @@index([sessionId])\n @@index([page])\n @@index([createdAt])\n @@index([country])\n @@index([deviceType])\n}\n\nmodel Session {\n id String @id @default(cuid())\n sessionId String @unique\n userId String?\n startedAt DateTime @default(now())\n endedAt DateTime?\n duration Int?\n pageViews PageView[]\n user User? @relation(fields: [userId], references: [id])\n workflowLogs WorkflowLog[]\n\n @@index([userId])\n @@index([startedAt])\n}\n\n// ============================================\n// STRIPE INTEGRATION MODELS\n// ============================================\n\n// Stripe Customer - polymorphic relation (User OR Organization)\nmodel StripeCustomer {\n id String @id @default(cuid())\n\n // ✅ Polymorphic relation - buď User (B2C) alebo Organization (B2B)\n userId String? @unique // B2C: Individuálny používateľ\n organizationId String? @unique // B2B: Organizácia\n\n stripeCustomerId String @unique\n email String\n name String?\n\n // Billing details (Stripe Address & Tax IDs)\n billingAddress Json? // Stripe Address object\n taxIds Json? // Stripe Tax IDs array (VAT, etc.)\n\n createdAt DateTime @default(now())\n updatedAt DateTime @updatedAt\n\n // Relations\n user User? @relation(fields: [userId], references: [id], onDelete: Cascade)\n organization Organization? @relation(fields: [organizationId], references: [id], onDelete: Cascade)\n subscriptions StripeSubscription[]\n payments StripePayment[]\n\n @@index([stripeCustomerId])\n @@index([userId])\n @@index([organizationId])\n @@index([email])\n}\n\n// Stripe Subscription - sledovanie predplatného (User alebo Organization)\nmodel StripeSubscription {\n id String @id @default(cuid())\n\n // ✅ FIX: customerId = FK na StripeCustomer.id (nie stripeCustomerId!)\n customerId String // FK na StripeCustomer.id\n stripeCustomerId String // Stripe ID (pre logging/redundancia)\n stripeSubscriptionId String @unique\n stripePriceId String\n stripeProductId String\n\n status SubscriptionStatus\n tier SubscriptionTier @default(FREE)\n billingInterval BillingInterval @default(MONTHLY) // Monthly or Yearly billing\n\n currentPeriodStart DateTime\n currentPeriodEnd DateTime\n cancelAtPeriodEnd Boolean @default(false)\n canceledAt DateTime?\n\n trialStart DateTime?\n trialEnd DateTime?\n\n // Additional fields\n quantity Int @default(1) // Počet seats (pre LAW_FIRM/ENTERPRISE tier)\n defaultPaymentMethodId String? // Stripe payment method ID\n\n metadata Json? // Dodatočné Stripe metadata\n createdAt DateTime @default(now())\n updatedAt DateTime @updatedAt\n\n // ✅ FIX: Relácia na customerId (interné ID), nie stripeCustomerId (Stripe ID)\n customer StripeCustomer @relation(fields: [customerId], references: [id], onDelete: Cascade)\n\n @@index([stripeSubscriptionId])\n @@index([customerId])\n @@index([stripeCustomerId])\n @@index([status])\n @@index([tier])\n @@index([billingInterval])\n @@index([currentPeriodEnd])\n}\n\n// Stripe Event - prevencia duplicitného spracovania webhookov (idempotencia)\nmodel StripeEvent {\n id String @id @default(cuid())\n eventId String @unique // Stripe event ID\n type String // Typ eventu (napr. 'checkout.session.completed')\n data Json // Úplné event data pre debugging\n processed Boolean @default(false)\n processedAt DateTime?\n error String? // Uloženie chýb pri spracovaní\n createdAt DateTime @default(now())\n\n @@index([eventId])\n @@index([processed])\n @@index([type])\n @@index([createdAt])\n}\n\n// Stripe Payment - sledovanie jednotlivých platieb (voliteľné, pre detailnú analytiku)\nmodel StripePayment {\n id String @id @default(cuid())\n\n // ✅ FIX: Pridaná relácia na StripeCustomer\n customerId String\n stripePaymentId String @unique\n stripeCustomerId String // Redundantné, ale OK pre logging\n\n amount Int // Suma v centoch\n currency String @default(\"eur\")\n status String // succeeded, failed, pending\n paymentMethod String? // card, sepa_debit, atď.\n description String?\n\n // Invoice info\n invoiceId String? // Stripe Invoice ID\n invoiceUrl String? // URL na faktúru\n\n metadata Json?\n createdAt DateTime @default(now())\n\n // ✅ Relácia na StripeCustomer\n customer StripeCustomer @relation(fields: [customerId], references: [id], onDelete: Cascade)\n\n @@index([stripePaymentId])\n @@index([customerId])\n @@index([stripeCustomerId])\n @@index([status])\n @@index([createdAt])\n}\n\n// ============================================\n// ENUMS\n// ============================================\n\nenum Role {\n USER\n ASSISTANT\n SYSTEM\n}\n\nenum FeedbackRating {\n LIKE\n DISLIKE\n NEUTRAL\n}\n\nenum ReferenceType {\n LAW\n CASE\n REGULATION\n DOCUMENT\n OTHER\n}\n\nenum ApprovalStatus {\n PENDING\n APPROVED\n REJECTED\n}\n\n// Stripe: Status predplatného\nenum SubscriptionStatus {\n ACTIVE // Predplatné je aktívne\n CANCELED // Predplatné zrušené\n INCOMPLETE // Iniciálna platba zlyhala\n INCOMPLETE_EXPIRED // Neúplné predplatné expirované\n PAST_DUE // Platba zlyhala, retry prebieha\n TRIALING // V skúšobnom období\n UNPAID // Platba zlyhala, žiadny retry\n PAUSED // Predplatné pozastavené (Stripe feature)\n}\n\n// Stripe: Tier predplatného\nenum SubscriptionTier {\n FREE // ✅ Bezplatný tier (10 správ)\n LAWYER // Právnik tier (1000 správ, €49/mes, 1 user)\n LAW_FIRM // Právna kancelária tier (5000 správ, €149/mes, až 5 users)\n ENTERPRISE // Enterprise tier (unlimited správ, unlimited users, custom price)\n}\n\n// Stripe: Billing interval\nenum BillingInterval {\n MONTHLY // Monthly billing\n YEARLY // Yearly billing (17% discount)\n}\n\n// Nový model pre logovanie admin akcií\nmodel AdminActionLog {\n id String @id @default(cuid())\n action String // APPROVE_USER, REJECT_USER, etc.\n targetUserId String // ID používateľa, na ktorého sa akcia vzťahuje\n adminEmail String // E-mail admina, ktorý vykonal akciu\n details Json? // Ďalšie detaily akcie (reason, notes, etc.)\n createdAt DateTime @default(now())\n\n @@index([action])\n @@index([targetUserId])\n @@index([adminEmail])\n @@index([createdAt])\n}\n\nmodel WorkflowLog {\n id String @id @default(cuid()) // Added @default(cuid())\n conversationId String\n messageId String?\n userId String? // Made optional to preserve logs when users are deleted\n sessionId String? // Removed @unique to allow multiple workflows per session\n startedAt DateTime @default(now())\n completedAt DateTime?\n status WorkflowStatus @default(IN_PROGRESS)\n totalDuration Int? // milliseconds\n error String?\n metadata Json?\n steps WorkflowStep[] // Renamed from WorkflowStep to steps (lowerCamel, plural)\n conversation Conversation @relation(fields: [conversationId], references: [id], onDelete: Cascade)\n user User? @relation(fields: [userId], references: [id], onDelete: SetNull)\n session Session? @relation(fields: [sessionId], references: [sessionId], onDelete: SetNull)\n answer Answer? @relation(fields: [messageId], references: [messageId])\n\n @@index([conversationId])\n @@index([userId])\n @@index([sessionId])\n @@index([startedAt])\n @@index([status])\n @@index([conversationId, startedAt]) // Composite index for timeline queries\n @@index([messageId])\n}\n\nmodel WorkflowStep {\n id String @id @default(cuid()) // Added @default(cuid())\n workflowId String\n stepName String // 'classify_question', 'rag_search', 'ai_processing', 'generate_response'\n stepType StepType\n startedAt DateTime @default(now())\n completedAt DateTime?\n duration Int? // milliseconds\n status StepStatus @default(IN_PROGRESS)\n inputData Json? // Sanitized input data\n outputData Json? // Sanitized output data\n error String?\n metadata Json? // Provider used, model used, tokens, costs, etc.\n workflow WorkflowLog @relation(fields: [workflowId], references: [id], onDelete: Cascade)\n\n @@index([workflowId])\n @@index([stepName])\n @@index([stepType])\n @@index([startedAt])\n @@index([status])\n @@index([workflowId, startedAt]) // Composite index for timeline queries\n}\n\nenum WorkflowStatus {\n IN_PROGRESS\n COMPLETED\n FAILED\n CANCELLED\n}\n\nenum StepStatus {\n IN_PROGRESS\n COMPLETED\n FAILED\n SKIPPED\n}\n\nenum StepType {\n CLASSIFICATION\n RAG_RETRIEVAL\n AI_PROCESSING\n RESPONSE_GENERATION\n DATA_PERSISTENCE\n ERROR_HANDLING\n}\n\n// Email verification tokens\nmodel VerificationToken {\n id String @id @default(cuid())\n userId String\n token String @unique\n expires DateTime\n createdAt DateTime @default(now())\n user User @relation(fields: [userId], references: [id], onDelete: Cascade)\n\n @@index([token])\n @@index([userId])\n @@index([expires])\n}\n\n// Password reset tokens\nmodel PasswordResetToken {\n id String @id @default(cuid())\n userId String\n token String @unique\n expires DateTime\n createdAt DateTime @default(now())\n user User @relation(fields: [userId], references: [id], onDelete: Cascade)\n\n @@index([token])\n @@index([userId])\n @@index([expires])\n}\n",
592
- "inlineSchemaHash": "4e15d6c5bf699f7af7fb79cf6c62acf2a20279262f0b10c498225eafe81c97b1",
591
+ "inlineSchema": "generator client {\n provider = \"prisma-client-js\"\n output = \"./generated/app\"\n binaryTargets = [\"native\", \"linux-musl-openssl-3.0.x\"]\n}\n\ndatasource db {\n provider = \"postgresql\"\n url = env(\"POSTGRES_PRISMA_URL\")\n directUrl = env(\"POSTGRES_URL_NON_POOLING\")\n}\n\nmodel Account {\n id String @id @default(cuid())\n userId String\n type String\n provider String\n providerAccountId String\n refresh_token String?\n access_token String?\n expires_at Int?\n token_type String?\n scope String?\n id_token String?\n session_state String?\n user User @relation(fields: [userId], references: [id], onDelete: Cascade)\n\n @@unique([provider, providerAccountId])\n}\n\nmodel User {\n id String @id @default(cuid())\n name String?\n email String? @unique\n emailVerified DateTime?\n image String?\n // Credentials authentication fields\n password String? // Hashed password pre credentials login\n createdAt DateTime @default(now())\n agreedToTerms Boolean @default(false)\n practiceArea String[]\n lawFirm String?\n yearsOfExperience Int?\n // Nové polia pre schvalovanie používateľov\n isApproved Boolean @default(false) // Či je používateľ schválený\n isRejected Boolean @default(false) // Či je používateľ zamietnutý\n approvedAt DateTime? // Kedy bol schválený\n rejectedAt DateTime? // Kedy bol zamietnutý\n approvedBy String? // ID admina, ktorý schválil\n rejectedBy String? // ID admina, ktorý zamietol\n rejectionReason String? // Dôvod zamietnutia\n // Nové polia pre tracking a žiadosť\n referralSource String? // Odkiaľ sa o nás dozvedel (Google, Facebook, LinkedIn, etc.)\n applicationText String? // Text žiadosti o prihlásenie\n // ✨ STRIPE FIELDS (B2C - len pre individuálnych používateľov)\n subscriptionTier SubscriptionTier @default(FREE) // Aktuálny subscription tier (cache)\n messageCount Int @default(10) // ✅ OPRAVENÉ z 100 na 10\n messageCountResetAt DateTime? // Kedy sa má resetovať message count\n // Relations\n approvalRequest UserApprovalRequest?\n stripeCustomer StripeCustomer? // ✨ B2C Stripe customer\n ownedOrganizations Organization[] @relation(\"OrganizationOwner\")\n organizationMembers OrganizationMember[]\n accounts Account[]\n answers Answer[]\n conversations Conversation[]\n feedbacks Feedback[]\n pageViews PageView[]\n sessions Session[]\n workflowLogs WorkflowLog[]\n verificationTokens VerificationToken[]\n passwordResetTokens PasswordResetToken[]\n}\n\n// Nový model pre žiadosti o schválenie\nmodel UserApprovalRequest {\n id String @id @default(cuid())\n userId String @unique\n status ApprovalStatus @default(PENDING)\n submittedAt DateTime @default(now())\n reviewedAt DateTime?\n reviewedBy String? // ID admina, ktorý preskúmal žiadosť\n notes String? // Poznámky admina\n user User @relation(fields: [userId], references: [id], onDelete: Cascade)\n\n @@index([status])\n @@index([submittedAt])\n @@index([reviewedAt])\n}\n\n// ============================================\n// ORGANIZATION MODELS (B2B)\n// ============================================\n\n// Organization = Právna kancelária / Firma\nmodel Organization {\n id String @id @default(cuid())\n name String // \"Advokátska kancelária XYZ s.r.o.\"\n\n // Billing & Legal Information\n companyNumber String? // IČO (Identifikačné číslo organizácie)\n vatNumber String? // IČ DPH (VAT Number)\n taxNumber String? // DIČ (Daňové identifikačné číslo)\n legalForm String? // \"s.r.o.\", \"a.s.\", \"v.o.s.\", \"k.s.\", \"SZČO\"\n billingEmail String\n\n // Address\n street String?\n city String?\n postalCode String?\n country String @default(\"SK\")\n\n // Subscription (B2B)\n subscriptionTier SubscriptionTier @default(FREE) // Tier organizácie (cache)\n messageCount Int @default(0) // Spoločný pool správ pre všetkých členov\n messageLimit Int @default(100) // Limit správ podľa tieru\n messageCountResetAt DateTime? // Kedy sa má resetovať pool\n\n // Settings\n isActive Boolean @default(true)\n maxMembers Int @default(5) // Max počet členov podľa tieru\n\n // Relations\n owner User @relation(\"OrganizationOwner\", fields: [ownerId], references: [id])\n ownerId String\n members OrganizationMember[]\n stripeCustomer StripeCustomer?\n\n createdAt DateTime @default(now())\n updatedAt DateTime @updatedAt\n\n @@index([ownerId])\n @@index([vatNumber])\n @@index([companyNumber])\n @@index([subscriptionTier])\n @@index([billingEmail])\n}\n\n// Many-to-Many: User ↔ Organization\nmodel OrganizationMember {\n id String @id @default(cuid())\n organizationId String\n userId String\n role MemberRole @default(MEMBER)\n joinedAt DateTime @default(now())\n\n organization Organization @relation(fields: [organizationId], references: [id], onDelete: Cascade)\n user User @relation(fields: [userId], references: [id], onDelete: Cascade)\n\n @@unique([organizationId, userId])\n @@index([organizationId])\n @@index([userId])\n @@index([role])\n}\n\nenum MemberRole {\n OWNER // Zakladateľ/vlastník, plný prístup k billing a settings\n ADMIN // Administrátor, môže pridávať/odoberať členov\n MEMBER // Bežný člen, len používa systém\n}\n\n// ZJEDNODUŠENÝ Conversation - len metadata, bez duplikátov\nmodel Conversation {\n id String @id @default(cuid())\n name String\n userId String\n isShareable Boolean @default(false)\n shareUrl String? @unique\n sharedAt DateTime?\n createdAt DateTime @default(now())\n updatedAt DateTime @updatedAt\n summary String?\n messagesSinceLastSummary Int? @default(0)\n // Relácie\n answers Answer[]\n user User @relation(fields: [userId], references: [id])\n workflowLogs WorkflowLog[]\n\n @@index([userId])\n @@index([createdAt])\n}\n\n// Hlavný model - všetky správy, bez duplikátov\nmodel Answer {\n id String @id @default(cuid())\n conversationId String\n messageId String @unique\n role Role\n content String\n question String?\n answer String?\n evaluation String?\n isWelcome Boolean @default(false)\n processingTime Int?\n model String?\n userId String?\n createdAt DateTime @default(now())\n updatedAt DateTime @updatedAt\n // Relácie\n conversation Conversation @relation(fields: [conversationId], references: [id], onDelete: Cascade)\n user User? @relation(fields: [userId], references: [id])\n feedback Feedback?\n references Reference[]\n metrics AnswerMetrics?\n WorkflowLog WorkflowLog[]\n files MessageFile[]\n\n @@index([conversationId])\n @@index([messageId])\n @@index([role])\n @@index([userId])\n @@index([createdAt])\n}\n\nmodel MessageFile {\n id String @id @default(cuid())\n answerId String\n fileName String\n fileType String\n base64Data String\n uploadedAt DateTime @default(now())\n answer Answer @relation(fields: [answerId], references: [id], onDelete: Cascade)\n\n @@index([answerId])\n @@index([fileType])\n @@index([uploadedAt])\n}\n\n// Nová tabuľka pre metriky odpovedí\nmodel AnswerMetrics {\n id String @id @default(cuid())\n answerId String @unique\n // Náklady\n apiCost Float? // Celkové náklady (ai_cost + rag_cost)\n aiCost Float? // Náklady len na AI provider volania\n ragCost Float? // Náklady na RAG operácie\n // Volania\n apiCalls Int? // Celkový počet volaní\n aiCalls Int? // Počet AI provider volaní\n ragCalls Int? // Počet RAG operácií\n // Čas a výkon\n apiDuration Int? // Doba spracovania v ms\n // AI Provider a Model\n aiProvider String? // Hlavný AI provider\n aiModel String? // Hlavný AI model\n aiProvidersUsed String[] // Všetky použité AI providery\n aiModelsUsed String[] // Všetky použité AI modely\n // Timestamps\n createdAt DateTime @default(now())\n updatedAt DateTime @updatedAt\n // Relations\n answer Answer @relation(fields: [answerId], references: [id], onDelete: Cascade)\n\n @@index([answerId])\n @@index([apiCost])\n @@index([apiDuration])\n @@index([aiProvider])\n @@index([createdAt])\n}\n\nmodel Feedback {\n id String @id @default(cuid())\n answerId String @unique\n rating FeedbackRating\n feedbackTypes String[]\n customFeedback String?\n userId String\n messageContent String?\n createdAt DateTime @default(now())\n answer Answer @relation(fields: [answerId], references: [id], onDelete: Cascade)\n user User @relation(fields: [userId], references: [id])\n\n @@index([rating])\n @@index([userId])\n @@index([answerId])\n}\n\n// ZJEDNODUŠENÝ Reference - len na Answer, bez duplikátov\nmodel Reference {\n id String @id @default(cuid())\n answerId String // Len answerId - konzistentné\n type ReferenceType\n title String\n reference String\n summary String?\n url String?\n uuid String?\n relevance Float?\n citationNumber Int?\n metadata Json?\n userId String?\n createdAt DateTime @default(now())\n // Relácie\n answer Answer @relation(fields: [answerId], references: [id], onDelete: Cascade)\n\n @@index([answerId])\n @@index([type])\n @@index([reference])\n @@index([userId])\n @@index([createdAt])\n}\n\nmodel PageView {\n id String @id @default(cuid())\n userId String?\n sessionId String?\n page String\n path String\n referrer String?\n userAgent String?\n ipAddress String?\n country String?\n city String?\n deviceType String?\n browser String?\n os String?\n screenSize String?\n language String?\n timeOnPage Int?\n isBounce Boolean @default(true)\n createdAt DateTime @default(now())\n session Session? @relation(fields: [sessionId], references: [sessionId], onDelete: SetNull)\n user User? @relation(fields: [userId], references: [id])\n\n @@index([userId])\n @@index([sessionId])\n @@index([page])\n @@index([createdAt])\n @@index([country])\n @@index([deviceType])\n}\n\nmodel Session {\n id String @id @default(cuid())\n sessionId String @unique\n userId String?\n startedAt DateTime @default(now())\n endedAt DateTime?\n duration Int?\n pageViews PageView[]\n user User? @relation(fields: [userId], references: [id])\n workflowLogs WorkflowLog[]\n\n @@index([userId])\n @@index([startedAt])\n}\n\n// ============================================\n// STRIPE INTEGRATION MODELS\n// ============================================\n\n// Stripe Customer - polymorphic relation (User OR Organization)\nmodel StripeCustomer {\n id String @id @default(cuid())\n\n // ✅ Polymorphic relation - buď User (B2C) alebo Organization (B2B)\n userId String? @unique // B2C: Individuálny používateľ\n organizationId String? @unique // B2B: Organizácia\n\n stripeCustomerId String @unique\n email String\n name String?\n\n // Billing details (Stripe Address & Tax IDs)\n billingAddress Json? // Stripe Address object\n taxIds Json? // Stripe Tax IDs array (VAT, etc.)\n\n createdAt DateTime @default(now())\n updatedAt DateTime @updatedAt\n\n // Relations\n user User? @relation(fields: [userId], references: [id], onDelete: Cascade)\n organization Organization? @relation(fields: [organizationId], references: [id], onDelete: Cascade)\n subscriptions StripeSubscription[]\n payments StripePayment[]\n\n @@index([stripeCustomerId])\n @@index([userId])\n @@index([organizationId])\n @@index([email])\n}\n\n// Stripe Subscription - sledovanie predplatného (User alebo Organization)\nmodel StripeSubscription {\n id String @id @default(cuid())\n\n // ✅ FIX: customerId = FK na StripeCustomer.id (nie stripeCustomerId!)\n customerId String // FK na StripeCustomer.id\n stripeCustomerId String // Stripe ID (pre logging/redundancia)\n stripeSubscriptionId String @unique\n stripePriceId String\n stripeProductId String\n\n status SubscriptionStatus\n tier SubscriptionTier @default(FREE)\n billingInterval BillingInterval @default(MONTHLY) // Monthly or Yearly billing\n\n currentPeriodStart DateTime\n currentPeriodEnd DateTime\n cancelAtPeriodEnd Boolean @default(false)\n canceledAt DateTime?\n\n trialStart DateTime?\n trialEnd DateTime?\n\n // Additional fields\n quantity Int @default(1) // Počet seats (pre LAW_FIRM/ENTERPRISE tier)\n defaultPaymentMethodId String? // Stripe payment method ID\n\n metadata Json? // Dodatočné Stripe metadata\n createdAt DateTime @default(now())\n updatedAt DateTime @updatedAt\n\n // ✅ FIX: Relácia na customerId (interné ID), nie stripeCustomerId (Stripe ID)\n customer StripeCustomer @relation(fields: [customerId], references: [id], onDelete: Cascade)\n\n @@index([stripeSubscriptionId])\n @@index([customerId])\n @@index([stripeCustomerId])\n @@index([status])\n @@index([tier])\n @@index([billingInterval])\n @@index([currentPeriodEnd])\n}\n\n// Stripe Event - prevencia duplicitného spracovania webhookov (idempotencia)\nmodel StripeEvent {\n id String @id @default(cuid())\n eventId String @unique // Stripe event ID\n type String // Typ eventu (napr. 'checkout.session.completed')\n data Json // Úplné event data pre debugging\n processed Boolean @default(false)\n processedAt DateTime?\n error String? // Uloženie chýb pri spracovaní\n createdAt DateTime @default(now())\n\n @@index([eventId])\n @@index([processed])\n @@index([type])\n @@index([createdAt])\n}\n\n// Stripe Payment - sledovanie jednotlivých platieb (voliteľné, pre detailnú analytiku)\nmodel StripePayment {\n id String @id @default(cuid())\n\n // ✅ FIX: Pridaná relácia na StripeCustomer\n customerId String\n stripePaymentId String @unique\n stripeCustomerId String // Redundantné, ale OK pre logging\n\n amount Int // Suma v centoch\n currency String @default(\"eur\")\n status String // succeeded, failed, pending\n paymentMethod String? // card, sepa_debit, atď.\n description String?\n\n // Invoice info\n invoiceId String? // Stripe Invoice ID\n invoiceUrl String? // URL na faktúru\n\n metadata Json?\n createdAt DateTime @default(now())\n\n // ✅ Relácia na StripeCustomer\n customer StripeCustomer @relation(fields: [customerId], references: [id], onDelete: Cascade)\n\n @@index([stripePaymentId])\n @@index([customerId])\n @@index([stripeCustomerId])\n @@index([status])\n @@index([createdAt])\n}\n\n// ============================================\n// ENUMS\n// ============================================\n\nenum Role {\n USER\n ASSISTANT\n SYSTEM\n}\n\nenum FeedbackRating {\n LIKE\n DISLIKE\n NEUTRAL\n}\n\nenum ReferenceType {\n LAW\n CASE\n REGULATION\n DOCUMENT\n OTHER\n}\n\nenum ApprovalStatus {\n PENDING\n APPROVED\n REJECTED\n}\n\n// Stripe: Status predplatného\nenum SubscriptionStatus {\n ACTIVE // Predplatné je aktívne\n CANCELED // Predplatné zrušené\n INCOMPLETE // Iniciálna platba zlyhala\n INCOMPLETE_EXPIRED // Neúplné predplatné expirované\n PAST_DUE // Platba zlyhala, retry prebieha\n TRIALING // V skúšobnom období\n UNPAID // Platba zlyhala, žiadny retry\n PAUSED // Predplatné pozastavené (Stripe feature)\n}\n\n// Stripe: Tier predplatného\nenum SubscriptionTier {\n FREE // Free tier (10 messages/month)\n LAWYER // Lawyer tier (1000 messages/month, €29/month, 1 user)\n LAWYER_PRO // Lawyer Pro tier (3000 messages/month, €49/month, 1 user) - RECOMMENDED\n LAW_FIRM // Law Firm tier (10000 messages/month, €129/month, up to 5 users)\n ENTERPRISE // Enterprise tier (unlimited messages, unlimited users, custom pricing)\n}\n\n// Stripe: Billing interval\nenum BillingInterval {\n MONTHLY // Monthly billing\n YEARLY // Yearly billing (17% discount)\n}\n\n// Nový model pre logovanie admin akcií\nmodel AdminActionLog {\n id String @id @default(cuid())\n action String // APPROVE_USER, REJECT_USER, etc.\n targetUserId String // ID používateľa, na ktorého sa akcia vzťahuje\n adminEmail String // E-mail admina, ktorý vykonal akciu\n details Json? // Ďalšie detaily akcie (reason, notes, etc.)\n createdAt DateTime @default(now())\n\n @@index([action])\n @@index([targetUserId])\n @@index([adminEmail])\n @@index([createdAt])\n}\n\nmodel WorkflowLog {\n id String @id @default(cuid()) // Added @default(cuid())\n conversationId String\n messageId String?\n userId String? // Made optional to preserve logs when users are deleted\n sessionId String? // Removed @unique to allow multiple workflows per session\n startedAt DateTime @default(now())\n completedAt DateTime?\n status WorkflowStatus @default(IN_PROGRESS)\n totalDuration Int? // milliseconds\n error String?\n metadata Json?\n steps WorkflowStep[] // Renamed from WorkflowStep to steps (lowerCamel, plural)\n conversation Conversation @relation(fields: [conversationId], references: [id], onDelete: Cascade)\n user User? @relation(fields: [userId], references: [id], onDelete: SetNull)\n session Session? @relation(fields: [sessionId], references: [sessionId], onDelete: SetNull)\n answer Answer? @relation(fields: [messageId], references: [messageId])\n\n @@index([conversationId])\n @@index([userId])\n @@index([sessionId])\n @@index([startedAt])\n @@index([status])\n @@index([conversationId, startedAt]) // Composite index for timeline queries\n @@index([messageId])\n}\n\nmodel WorkflowStep {\n id String @id @default(cuid()) // Added @default(cuid())\n workflowId String\n stepName String // 'classify_question', 'rag_search', 'ai_processing', 'generate_response'\n stepType StepType\n startedAt DateTime @default(now())\n completedAt DateTime?\n duration Int? // milliseconds\n status StepStatus @default(IN_PROGRESS)\n inputData Json? // Sanitized input data\n outputData Json? // Sanitized output data\n error String?\n metadata Json? // Provider used, model used, tokens, costs, etc.\n workflow WorkflowLog @relation(fields: [workflowId], references: [id], onDelete: Cascade)\n\n @@index([workflowId])\n @@index([stepName])\n @@index([stepType])\n @@index([startedAt])\n @@index([status])\n @@index([workflowId, startedAt]) // Composite index for timeline queries\n}\n\nenum WorkflowStatus {\n IN_PROGRESS\n COMPLETED\n FAILED\n CANCELLED\n}\n\nenum StepStatus {\n IN_PROGRESS\n COMPLETED\n FAILED\n SKIPPED\n}\n\nenum StepType {\n CLASSIFICATION\n RAG_RETRIEVAL\n AI_PROCESSING\n RESPONSE_GENERATION\n DATA_PERSISTENCE\n ERROR_HANDLING\n}\n\n// Email verification tokens\nmodel VerificationToken {\n id String @id @default(cuid())\n userId String\n token String @unique\n expires DateTime\n createdAt DateTime @default(now())\n user User @relation(fields: [userId], references: [id], onDelete: Cascade)\n\n @@index([token])\n @@index([userId])\n @@index([expires])\n}\n\n// Password reset tokens\nmodel PasswordResetToken {\n id String @id @default(cuid())\n userId String\n token String @unique\n expires DateTime\n createdAt DateTime @default(now())\n user User @relation(fields: [userId], references: [id], onDelete: Cascade)\n\n @@index([token])\n @@index([userId])\n @@index([expires])\n}\n",
592
+ "inlineSchemaHash": "d03d75c2776b8e1186d1a3f4a22257fce76ce70ad533c688c48d16e94b2fa34b",
593
593
  "copyEngine": true
594
594
  }
595
595
  config.dirname = '/'
@@ -0,0 +1,2 @@
1
+ -- AlterEnum
2
+ ALTER TYPE "SubscriptionTier" ADD VALUE 'LAWYER_PRO';