2024-06-18 01:50:39 -07:00
import { Container } from 'typedi' ;
import { v4 as uuid } from 'uuid' ;
import { EntityNotFoundError } from '@n8n/typeorm' ;
2024-05-17 01:53:15 -07:00
2024-06-18 01:50:39 -07:00
import { Reset } from '@/commands/ldap/reset' ;
2024-05-17 01:53:15 -07:00
import { LoadNodesAndCredentials } from '@/LoadNodesAndCredentials' ;
import { InternalHooks } from '@/InternalHooks' ;
2024-06-18 01:50:39 -07:00
import { WorkflowRepository } from '@db/repositories/workflow.repository' ;
import { CredentialsRepository } from '@db/repositories/credentials.repository' ;
import { SharedWorkflowRepository } from '@db/repositories/sharedWorkflow.repository' ;
import { SharedCredentialsRepository } from '@db/repositories/sharedCredentials.repository' ;
import { getLdapSynchronizations , saveLdapSynchronization } from '@/Ldap/helpers' ;
import { LdapService } from '@/Ldap/ldap.service' ;
import { Push } from '@/push' ;
import { Telemetry } from '@/telemetry' ;
import { setupTestCommand } from '@test-integration/utils/testCommand' ;
import { mockInstance } from '../../../shared/mocking' ;
2024-05-17 01:53:15 -07:00
import { createLdapUser , createMember , getUserById } from '../../shared/db/users' ;
import { createWorkflow } from '../../shared/db/workflows' ;
import { randomCredentialPayload } from '../../shared/random' ;
import { saveCredential } from '../../shared/db/credentials' ;
import { createLdapConfig } from '../../shared/ldap' ;
2024-06-18 01:50:39 -07:00
import { createTeamProject , findProject , getPersonalProject } from '../../shared/db/projects' ;
2024-06-12 06:05:43 -07:00
mockInstance ( Telemetry ) ;
2024-05-17 01:53:15 -07:00
2024-06-18 01:50:39 -07:00
mockInstance ( Push ) ;
mockInstance ( InternalHooks ) ;
mockInstance ( LoadNodesAndCredentials ) ;
const command = setupTestCommand ( Reset ) ;
2024-05-17 01:53:15 -07:00
test ( 'fails if neither `--userId` nor `--projectId` nor `--deleteWorkflowsAndCredentials` is passed' , async ( ) = > {
2024-06-18 01:50:39 -07:00
await expect ( command . run ( ) ) . rejects . toThrowError (
2024-05-17 01:53:15 -07:00
'You must use exactly one of `--userId`, `--projectId` or `--deleteWorkflowsAndCredentials`.' ,
) ;
} ) ;
test . each ( [
[ ` --userId= ${ uuid ( ) } ` , ` --projectId= ${ uuid ( ) } ` , '--deleteWorkflowsAndCredentials' ] ,
[ ` --userId= ${ uuid ( ) } ` , ` --projectId= ${ uuid ( ) } ` ] ,
[ ` --userId= ${ uuid ( ) } ` , '--deleteWorkflowsAndCredentials' ] ,
[ '--deleteWorkflowsAndCredentials' , ` --projectId= ${ uuid ( ) } ` ] ,
] ) (
'fails if more than one of `--userId`, `--projectId`, `--deleteWorkflowsAndCredentials` are passed' ,
async ( . . . argv ) = > {
2024-06-18 01:50:39 -07:00
await expect ( command . run ( argv ) ) . rejects . toThrowError (
2024-05-17 01:53:15 -07:00
'You must use exactly one of `--userId`, `--projectId` or `--deleteWorkflowsAndCredentials`.' ,
) ;
} ,
) ;
describe ( '--deleteWorkflowsAndCredentials' , ( ) = > {
test ( 'deletes personal projects, workflows and credentials owned by LDAP managed users' , async ( ) = > {
//
// ARRANGE
//
const member = await createLdapUser ( { role : 'global:member' } , uuid ( ) ) ;
const memberProject = await getPersonalProject ( member ) ;
const workflow = await createWorkflow ( { } , member ) ;
const credential = await saveCredential ( randomCredentialPayload ( ) , {
user : member ,
role : 'credential:owner' ,
} ) ;
const normalMember = await createMember ( ) ;
const workflow2 = await createWorkflow ( { } , normalMember ) ;
const credential2 = await saveCredential ( randomCredentialPayload ( ) , {
user : normalMember ,
role : 'credential:owner' ,
} ) ;
//
// ACT
//
2024-06-18 01:50:39 -07:00
await command . run ( [ '--deleteWorkflowsAndCredentials' ] ) ;
2024-05-17 01:53:15 -07:00
//
// ASSERT
//
// LDAP user is deleted
await expect ( getUserById ( member . id ) ) . rejects . toThrowError ( EntityNotFoundError ) ;
await expect ( findProject ( memberProject . id ) ) . rejects . toThrowError ( EntityNotFoundError ) ;
await expect (
Container . get ( WorkflowRepository ) . findOneBy ( { id : workflow.id } ) ,
) . resolves . toBeNull ( ) ;
await expect (
Container . get ( CredentialsRepository ) . findOneBy ( { id : credential.id } ) ,
) . resolves . toBeNull ( ) ;
// Non LDAP user is not deleted
await expect ( getUserById ( normalMember . id ) ) . resolves . not . toThrowError ( ) ;
await expect (
Container . get ( WorkflowRepository ) . findOneBy ( { id : workflow2.id } ) ,
) . resolves . not . toBeNull ( ) ;
await expect (
Container . get ( CredentialsRepository ) . findOneBy ( { id : credential2.id } ) ,
) . resolves . not . toBeNull ( ) ;
} ) ;
test ( 'deletes the LDAP sync history' , async ( ) = > {
//
// ARRANGE
//
await saveLdapSynchronization ( {
created : 1 ,
disabled : 1 ,
scanned : 1 ,
updated : 1 ,
endedAt : new Date ( ) ,
startedAt : new Date ( ) ,
error : '' ,
runMode : 'dry' ,
status : 'success' ,
} ) ;
//
// ACT
//
2024-06-18 01:50:39 -07:00
await command . run ( [ '--deleteWorkflowsAndCredentials' ] ) ;
2024-05-17 01:53:15 -07:00
//
// ASSERT
//
await expect ( getLdapSynchronizations ( 0 , 10 ) ) . resolves . toHaveLength ( 0 ) ;
} ) ;
test ( 'resets LDAP settings' , async ( ) = > {
//
// ARRANGE
//
await createLdapConfig ( ) ;
await expect ( Container . get ( LdapService ) . loadConfig ( ) ) . resolves . toMatchObject ( {
loginEnabled : true ,
} ) ;
//
// ACT
//
2024-06-18 01:50:39 -07:00
await command . run ( [ '--deleteWorkflowsAndCredentials' ] ) ;
2024-05-17 01:53:15 -07:00
//
// ASSERT
//
await expect ( Container . get ( LdapService ) . loadConfig ( ) ) . resolves . toMatchObject ( {
loginEnabled : false ,
} ) ;
} ) ;
} ) ;
describe ( '--userId' , ( ) = > {
test ( 'fails if the user does not exist' , async ( ) = > {
const userId = uuid ( ) ;
2024-06-18 01:50:39 -07:00
await expect ( command . run ( [ ` --userId= ${ userId } ` ] ) ) . rejects . toThrowError (
2024-05-17 01:53:15 -07:00
` Could not find the user with the ID ${ userId } or their personalProject. ` ,
) ;
} ) ;
test ( 'fails if the user to migrate to is also an LDAP user' , async ( ) = > {
//
// ARRANGE
//
const member = await createLdapUser ( { role : 'global:member' } , uuid ( ) ) ;
2024-06-18 01:50:39 -07:00
await expect ( command . run ( [ ` --userId= ${ member . id } ` ] ) ) . rejects . toThrowError (
2024-05-17 01:53:15 -07:00
` Can't migrate workflows and credentials to the user with the ID ${ member . id } . That user was created via LDAP and will be deleted as well. ` ,
) ;
} ) ;
test ( "transfers all workflows and credentials to the user's personal project" , async ( ) = > {
//
// ARRANGE
//
const member = await createLdapUser ( { role : 'global:member' } , uuid ( ) ) ;
const memberProject = await getPersonalProject ( member ) ;
const workflow = await createWorkflow ( { } , member ) ;
const credential = await saveCredential ( randomCredentialPayload ( ) , {
user : member ,
role : 'credential:owner' ,
} ) ;
const normalMember = await createMember ( ) ;
const normalMemberProject = await getPersonalProject ( normalMember ) ;
const workflow2 = await createWorkflow ( { } , normalMember ) ;
const credential2 = await saveCredential ( randomCredentialPayload ( ) , {
user : normalMember ,
role : 'credential:owner' ,
} ) ;
//
// ACT
//
2024-06-18 01:50:39 -07:00
await command . run ( [ ` --userId= ${ normalMember . id } ` ] ) ;
2024-05-17 01:53:15 -07:00
//
// ASSERT
//
// LDAP user is deleted
await expect ( getUserById ( member . id ) ) . rejects . toThrowError ( EntityNotFoundError ) ;
await expect ( findProject ( memberProject . id ) ) . rejects . toThrowError ( EntityNotFoundError ) ;
// Their workflow and credential have been migrated to the normal user.
await expect (
Container . get ( SharedWorkflowRepository ) . findOneBy ( {
workflowId : workflow.id ,
projectId : normalMemberProject.id ,
} ) ,
) . resolves . not . toBeNull ( ) ;
await expect (
Container . get ( SharedCredentialsRepository ) . findOneBy ( {
credentialsId : credential.id ,
projectId : normalMemberProject.id ,
} ) ,
) . resolves . not . toBeNull ( ) ;
// Non LDAP user is not deleted
await expect ( getUserById ( normalMember . id ) ) . resolves . not . toThrowError ( ) ;
await expect (
Container . get ( WorkflowRepository ) . findOneBy ( { id : workflow2.id } ) ,
) . resolves . not . toBeNull ( ) ;
await expect (
Container . get ( CredentialsRepository ) . findOneBy ( { id : credential2.id } ) ,
) . resolves . not . toBeNull ( ) ;
} ) ;
} ) ;
describe ( '--projectId' , ( ) = > {
test ( 'fails if the project does not exist' , async ( ) = > {
const projectId = uuid ( ) ;
2024-06-18 01:50:39 -07:00
await expect ( command . run ( [ ` --projectId= ${ projectId } ` ] ) ) . rejects . toThrowError (
2024-05-17 01:53:15 -07:00
` Could not find the project with the ID ${ projectId } . ` ,
) ;
} ) ;
test ( 'fails if the user to migrate to is also an LDAP user' , async ( ) = > {
//
// ARRANGE
//
const member = await createLdapUser ( { role : 'global:member' } , uuid ( ) ) ;
const memberProject = await getPersonalProject ( member ) ;
2024-06-18 01:50:39 -07:00
await expect ( command . run ( [ ` --projectId= ${ memberProject . id } ` ] ) ) . rejects . toThrowError (
2024-05-17 01:53:15 -07:00
` Can't migrate workflows and credentials to the project with the ID ${ memberProject . id } . That project is a personal project belonging to a user that was created via LDAP and will be deleted as well. ` ,
) ;
} ) ;
test ( 'transfers all workflows and credentials to a personal project' , async ( ) = > {
//
// ARRANGE
//
const member = await createLdapUser ( { role : 'global:member' } , uuid ( ) ) ;
const memberProject = await getPersonalProject ( member ) ;
const workflow = await createWorkflow ( { } , member ) ;
const credential = await saveCredential ( randomCredentialPayload ( ) , {
user : member ,
role : 'credential:owner' ,
} ) ;
const normalMember = await createMember ( ) ;
const normalMemberProject = await getPersonalProject ( normalMember ) ;
const workflow2 = await createWorkflow ( { } , normalMember ) ;
const credential2 = await saveCredential ( randomCredentialPayload ( ) , {
user : normalMember ,
role : 'credential:owner' ,
} ) ;
//
// ACT
//
2024-06-18 01:50:39 -07:00
await command . run ( [ ` --projectId= ${ normalMemberProject . id } ` ] ) ;
2024-05-17 01:53:15 -07:00
//
// ASSERT
//
// LDAP user is deleted
await expect ( getUserById ( member . id ) ) . rejects . toThrowError ( EntityNotFoundError ) ;
await expect ( findProject ( memberProject . id ) ) . rejects . toThrowError ( EntityNotFoundError ) ;
// Their workflow and credential have been migrated to the normal user.
await expect (
Container . get ( SharedWorkflowRepository ) . findOneBy ( {
workflowId : workflow.id ,
projectId : normalMemberProject.id ,
} ) ,
) . resolves . not . toBeNull ( ) ;
await expect (
Container . get ( SharedCredentialsRepository ) . findOneBy ( {
credentialsId : credential.id ,
projectId : normalMemberProject.id ,
} ) ,
) . resolves . not . toBeNull ( ) ;
// Non LDAP user is not deleted
await expect ( getUserById ( normalMember . id ) ) . resolves . not . toThrowError ( ) ;
await expect (
Container . get ( WorkflowRepository ) . findOneBy ( { id : workflow2.id } ) ,
) . resolves . not . toBeNull ( ) ;
await expect (
Container . get ( CredentialsRepository ) . findOneBy ( { id : credential2.id } ) ,
) . resolves . not . toBeNull ( ) ;
} ) ;
test ( 'transfers all workflows and credentials to a team project' , async ( ) = > {
//
// ARRANGE
//
const member = await createLdapUser ( { role : 'global:member' } , uuid ( ) ) ;
const memberProject = await getPersonalProject ( member ) ;
const workflow = await createWorkflow ( { } , member ) ;
const credential = await saveCredential ( randomCredentialPayload ( ) , {
user : member ,
role : 'credential:owner' ,
} ) ;
const normalMember = await createMember ( ) ;
const workflow2 = await createWorkflow ( { } , normalMember ) ;
const credential2 = await saveCredential ( randomCredentialPayload ( ) , {
user : normalMember ,
role : 'credential:owner' ,
} ) ;
const teamProject = await createTeamProject ( ) ;
//
// ACT
//
2024-06-18 01:50:39 -07:00
await command . run ( [ ` --projectId= ${ teamProject . id } ` ] ) ;
2024-05-17 01:53:15 -07:00
//
// ASSERT
//
// LDAP user is deleted
await expect ( getUserById ( member . id ) ) . rejects . toThrowError ( EntityNotFoundError ) ;
await expect ( findProject ( memberProject . id ) ) . rejects . toThrowError ( EntityNotFoundError ) ;
// Their workflow and credential have been migrated to the team project.
await expect (
Container . get ( SharedWorkflowRepository ) . findOneBy ( {
workflowId : workflow.id ,
projectId : teamProject.id ,
} ) ,
) . resolves . not . toBeNull ( ) ;
await expect (
Container . get ( SharedCredentialsRepository ) . findOneBy ( {
credentialsId : credential.id ,
projectId : teamProject.id ,
} ) ,
) . resolves . not . toBeNull ( ) ;
// Non LDAP user is not deleted
await expect ( getUserById ( normalMember . id ) ) . resolves . not . toThrowError ( ) ;
await expect (
Container . get ( WorkflowRepository ) . findOneBy ( { id : workflow2.id } ) ,
) . resolves . not . toBeNull ( ) ;
await expect (
Container . get ( CredentialsRepository ) . findOneBy ( { id : credential2.id } ) ,
) . resolves . not . toBeNull ( ) ;
} ) ;
} ) ;