tenant.service.ts 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  1. import {
  2. BadRequestException,
  3. ForbiddenException,
  4. Injectable,
  5. NotFoundException,
  6. } from '@nestjs/common';
  7. import { InjectRepository } from '@nestjs/typeorm';
  8. import { Repository } from 'typeorm';
  9. import { Tenant } from './tenant.entity';
  10. import { TenantSetting } from './tenant-setting.entity';
  11. import { TenantMember } from './tenant-member.entity';
  12. @Injectable()
  13. export class TenantService {
  14. public static readonly DEFAULT_TENANT_NAME = 'Default';
  15. constructor(
  16. @InjectRepository(Tenant)
  17. private readonly tenantRepository: Repository<Tenant>,
  18. @InjectRepository(TenantSetting)
  19. private readonly tenantSettingRepository: Repository<TenantSetting>,
  20. @InjectRepository(TenantMember)
  21. private readonly tenantMemberRepository: Repository<TenantMember>,
  22. ) { }
  23. async findAll(): Promise<Tenant[]> {
  24. return this.tenantRepository.find({
  25. relations: ['members', 'members.user'],
  26. order: { createdAt: 'ASC' }
  27. });
  28. }
  29. async findById(id: string): Promise<Tenant> {
  30. const tenant = await this.tenantRepository.findOneBy({ id });
  31. if (!tenant) throw new NotFoundException(`Tenant ${id} not found`);
  32. return tenant;
  33. }
  34. async findByName(name: string): Promise<Tenant | null> {
  35. return this.tenantRepository.findOneBy({ name });
  36. }
  37. async create(name: string, domain?: string, parentId?: string, isSystem: boolean = false): Promise<Tenant> {
  38. const existing = await this.findByName(name);
  39. if (existing) throw new BadRequestException(`Tenant name "${name}" already exists`);
  40. const tenant = this.tenantRepository.create({ name, domain, parentId, isSystem });
  41. const saved = await this.tenantRepository.save(tenant);
  42. // Auto-create default TenantSettings
  43. const setting = this.tenantSettingRepository.create({ tenantId: saved.id });
  44. await this.tenantSettingRepository.save(setting);
  45. return saved;
  46. }
  47. async update(id: string, data: Partial<Tenant>): Promise<Tenant> {
  48. const tenant = await this.findById(id);
  49. if (tenant.isSystem) {
  50. throw new ForbiddenException(`Cannot modify a system organization`);
  51. }
  52. await this.tenantRepository.save({ ...tenant, ...data });
  53. return this.findById(id);
  54. }
  55. async remove(id: string): Promise<void> {
  56. const tenant = await this.findById(id);
  57. if (tenant.isSystem) {
  58. throw new ForbiddenException(`Cannot delete a system organization`);
  59. }
  60. await this.tenantRepository.delete(id);
  61. }
  62. async getSettings(tenantId: string): Promise<TenantSetting> {
  63. let setting = await this.tenantSettingRepository.findOneBy({ tenantId });
  64. if (!setting) {
  65. // Defensive: Check if tenant actually exists before creating settings
  66. // to avoid FOREIGN KEY constraint failure if tenantId is invalid/legacy
  67. const tenantExists = await this.tenantRepository.findOneBy({ id: tenantId });
  68. if (!tenantExists) {
  69. console.warn(`[TenantService] Attempted to get settings for non-existent tenant: ${tenantId}`);
  70. // Return a transient default object without saving to DB
  71. return this.tenantSettingRepository.create({ tenantId });
  72. }
  73. setting = this.tenantSettingRepository.create({ tenantId });
  74. setting = await this.tenantSettingRepository.save(setting);
  75. }
  76. return setting;
  77. }
  78. async updateSettings(tenantId: string, data: Partial<TenantSetting>): Promise<TenantSetting> {
  79. let setting = await this.tenantSettingRepository.findOneBy({ tenantId });
  80. if (!setting) {
  81. setting = this.tenantSettingRepository.create({ tenantId, ...data });
  82. } else {
  83. if (data.enabledModelIds) {
  84. if (setting.selectedLLMId && !data.enabledModelIds.includes(setting.selectedLLMId)) {
  85. data.selectedLLMId = null as any;
  86. }
  87. if (setting.selectedEmbeddingId && !data.enabledModelIds.includes(setting.selectedEmbeddingId)) {
  88. data.selectedEmbeddingId = null as any;
  89. }
  90. if (setting.selectedRerankId && !data.enabledModelIds.includes(setting.selectedRerankId)) {
  91. data.selectedRerankId = null as any;
  92. }
  93. }
  94. Object.assign(setting, data);
  95. }
  96. return this.tenantSettingRepository.save(setting);
  97. }
  98. async updateMemberRole(tenantId: string, userId: string, role: string): Promise<TenantMember> {
  99. const existing = await this.tenantMemberRepository.findOneBy({ tenantId, userId });
  100. if (!existing) {
  101. throw new ForbiddenException(`Member not found in this organization`);
  102. }
  103. existing.role = role as any;
  104. return this.tenantMemberRepository.save(existing);
  105. }
  106. async addMember(tenantId: string, userId: string, role: string = 'USER'): Promise<TenantMember> {
  107. const tenant = await this.findById(tenantId);
  108. if (tenant.isSystem) {
  109. throw new ForbiddenException(`Cannot manually bind members to a system organization`);
  110. }
  111. const existing = await this.tenantMemberRepository.findOneBy({ tenantId, userId });
  112. if (existing) {
  113. existing.role = role as any;
  114. return this.tenantMemberRepository.save(existing);
  115. }
  116. const member = this.tenantMemberRepository.create({ tenantId, userId, role: role as any });
  117. return this.tenantMemberRepository.save(member);
  118. }
  119. async removeMember(tenantId: string, userId: string): Promise<void> {
  120. await this.tenantMemberRepository.delete({ tenantId, userId });
  121. }
  122. async getMembers(tenantId: string, page?: number, limit?: number): Promise<{ data: TenantMember[]; total: number }> {
  123. const queryBuilder = this.tenantMemberRepository.createQueryBuilder('member')
  124. .leftJoinAndSelect('member.user', 'user')
  125. .where('member.tenantId = :tenantId', { tenantId })
  126. .select(['member', 'user.id', 'user.username', 'user.displayName', 'user.isAdmin']);
  127. if (page && limit) {
  128. const [data, total] = await queryBuilder
  129. .skip((page - 1) * limit)
  130. .take(limit)
  131. .getManyAndCount();
  132. return { data, total };
  133. }
  134. const [data, total] = await queryBuilder.getManyAndCount();
  135. return { data, total };
  136. }
  137. /**
  138. * Ensure a "Default" tenant exists for data migration purposes.
  139. * Called during app bootstrap.
  140. */
  141. async ensureDefaultTenant(): Promise<Tenant> {
  142. let defaultTenant = await this.findByName(TenantService.DEFAULT_TENANT_NAME);
  143. if (!defaultTenant) {
  144. defaultTenant = await this.create(TenantService.DEFAULT_TENANT_NAME, 'default.localhost', undefined, true);
  145. } else if (!defaultTenant.isSystem) {
  146. defaultTenant.isSystem = true;
  147. await this.tenantRepository.save(defaultTenant);
  148. }
  149. return defaultTenant;
  150. }
  151. }