| 
 | 1 | +import { Test } from '@nestjs/testing'  | 
 | 2 | +import { SecretManagerServiceClient } from '@google-cloud/secret-manager'  | 
 | 3 | +import { SecretLoaderService } from './secretloader.service'  | 
 | 4 | +import { SecretLoadException } from './exceptions'  | 
 | 5 | +import { SECRET_MANAGER_CLIENT_TOKEN } from './constants'  | 
 | 6 | + | 
 | 7 | +jest.mock('@google-cloud/secret-manager')  | 
 | 8 | +describe('SecretLoaderService', () => {  | 
 | 9 | +  describe('static', () => {  | 
 | 10 | +    it('Can load 1 secret statically', async () => {  | 
 | 11 | +      const client = new SecretManagerServiceClient()  | 
 | 12 | +      ;(client.accessSecretVersion as jest.Mock).mockImplementationOnce(({ name }) => [  | 
 | 13 | +        {  | 
 | 14 | +          payload: { data: name },  | 
 | 15 | +        },  | 
 | 16 | +      ])  | 
 | 17 | + | 
 | 18 | +      const secrets = ['project/some-project/secrets/some-secret/versions/latest']  | 
 | 19 | +      const values = await SecretLoaderService.staticLoadSecrets(client, secrets)  | 
 | 20 | +      expect(client.accessSecretVersion).toBeCalledTimes(1)  | 
 | 21 | +      expect(values).toEqual({  | 
 | 22 | +        'project/some-project/secrets/some-secret/versions/latest': 'project/some-project/secrets/some-secret/versions/latest',  | 
 | 23 | +      })  | 
 | 24 | +    })  | 
 | 25 | + | 
 | 26 | +    it('Can load 0 secrets statically', async () => {  | 
 | 27 | +      const client = new SecretManagerServiceClient()  | 
 | 28 | +      ;(client.accessSecretVersion as jest.Mock).mockImplementationOnce(({ name }) => [  | 
 | 29 | +        {  | 
 | 30 | +          payload: { data: name },  | 
 | 31 | +        },  | 
 | 32 | +      ])  | 
 | 33 | + | 
 | 34 | +      const secrets: string[] = []  | 
 | 35 | +      const values = await SecretLoaderService.staticLoadSecrets(client, secrets)  | 
 | 36 | +      expect(client.accessSecretVersion).toBeCalledTimes(0)  | 
 | 37 | +      expect(values).toEqual({})  | 
 | 38 | +    })  | 
 | 39 | + | 
 | 40 | +    it('Can load 2 secrets statically', async () => {  | 
 | 41 | +      const client = new SecretManagerServiceClient()  | 
 | 42 | +      ;(client.accessSecretVersion as jest.Mock).mockImplementation(({ name }) => [  | 
 | 43 | +        {  | 
 | 44 | +          payload: { data: name },  | 
 | 45 | +        },  | 
 | 46 | +      ])  | 
 | 47 | + | 
 | 48 | +      const secrets: string[] = ['project/some-project/secrets/some-secret/versions/latest', 'project/some-project/secrets/some-secret/versions/1']  | 
 | 49 | +      const values = await SecretLoaderService.staticLoadSecrets(client, secrets)  | 
 | 50 | +      expect(client.accessSecretVersion).toBeCalledTimes(2)  | 
 | 51 | +      expect(values).toEqual({  | 
 | 52 | +        'project/some-project/secrets/some-secret/versions/latest': 'project/some-project/secrets/some-secret/versions/latest',  | 
 | 53 | +        'project/some-project/secrets/some-secret/versions/1': 'project/some-project/secrets/some-secret/versions/1',  | 
 | 54 | +      })  | 
 | 55 | +    })  | 
 | 56 | + | 
 | 57 | +    it('Overrides result object if the same secret is loaded twice', async () => {  | 
 | 58 | +      const client = new SecretManagerServiceClient()  | 
 | 59 | +      ;(client.accessSecretVersion as jest.Mock).mockImplementation(({ name }) => [  | 
 | 60 | +        {  | 
 | 61 | +          payload: { data: name },  | 
 | 62 | +        },  | 
 | 63 | +      ])  | 
 | 64 | + | 
 | 65 | +      const secrets: string[] = [  | 
 | 66 | +        'project/some-project/secrets/some-secret/versions/latest',  | 
 | 67 | +        'project/some-project/secrets/some-secret/versions/latest',  | 
 | 68 | +      ]  | 
 | 69 | +      const values = await SecretLoaderService.staticLoadSecrets(client, secrets)  | 
 | 70 | +      expect(client.accessSecretVersion).toBeCalledTimes(2)  | 
 | 71 | +      expect(values).toEqual({  | 
 | 72 | +        'project/some-project/secrets/some-secret/versions/latest': 'project/some-project/secrets/some-secret/versions/latest',  | 
 | 73 | +      })  | 
 | 74 | +    })  | 
 | 75 | + | 
 | 76 | +    it('Throws an error if secret load fails', async () => {  | 
 | 77 | +      const client = new SecretManagerServiceClient()  | 
 | 78 | +      ;(client.accessSecretVersion as jest.Mock).mockImplementation(() => {  | 
 | 79 | +        throw new SecretLoadException('Load failed')  | 
 | 80 | +      })  | 
 | 81 | + | 
 | 82 | +      const secrets: string[] = ['project/some-project/secrets/some-secret/versions/latest']  | 
 | 83 | +      await expect(SecretLoaderService.staticLoadSecrets(client, secrets)).rejects.toThrow(SecretLoadException)  | 
 | 84 | +    })  | 
 | 85 | +  })  | 
 | 86 | + | 
 | 87 | +  describe('dynamic', () => {  | 
 | 88 | +    let secretLoadService: SecretLoaderService  | 
 | 89 | +    let client: SecretManagerServiceClient  | 
 | 90 | +    beforeEach(async () => {  | 
 | 91 | +      client = new SecretManagerServiceClient()  | 
 | 92 | +      ;(client.accessSecretVersion as jest.Mock).mockImplementation(({ name }) => [  | 
 | 93 | +        {  | 
 | 94 | +          payload: { data: name },  | 
 | 95 | +        },  | 
 | 96 | +      ])  | 
 | 97 | +      const moduleRef = await Test.createTestingModule({  | 
 | 98 | +        providers: [  | 
 | 99 | +          SecretLoaderService,  | 
 | 100 | +          {  | 
 | 101 | +            provide: SECRET_MANAGER_CLIENT_TOKEN,  | 
 | 102 | +            useValue: client,  | 
 | 103 | +          },  | 
 | 104 | +        ],  | 
 | 105 | +      }).compile()  | 
 | 106 | +      secretLoadService = moduleRef.get<SecretLoaderService>(SecretLoaderService)  | 
 | 107 | +    })  | 
 | 108 | + | 
 | 109 | +    it('Can load a secret dynamically', async () => {  | 
 | 110 | +      const secret = 'project/some-project/secrets/some-secret/versions/latest'  | 
 | 111 | +      const value = await secretLoadService.loadSecret(secret)  | 
 | 112 | +      expect(secret).toEqual(value)  | 
 | 113 | +      expect(client.accessSecretVersion).toHaveBeenCalledTimes(1)  | 
 | 114 | +    })  | 
 | 115 | +  })  | 
 | 116 | +})  | 
0 commit comments