mirror of
https://github.com/immich-app/immich.git
synced 2025-01-09 21:36:46 +01:00
5aa658de59
Modify Access repository, to evaluate `asset` permissions in bulk. Queries have been validated to match what they currently generate for single ids. Queries: * `asset` album access: ```sql -- Before SELECT 1 AS "row_exists" FROM (SELECT 1 AS dummy_column) "dummy_table" WHERE EXISTS ( SELECT 1 FROM "albums" "AlbumEntity" LEFT JOIN "albums_assets_assets" "AlbumEntity_AlbumEntity__AlbumEntity_assets" ON "AlbumEntity_AlbumEntity__AlbumEntity_assets"."albumsId"="AlbumEntity"."id" LEFT JOIN "assets" "AlbumEntity__AlbumEntity_assets" ON "AlbumEntity__AlbumEntity_assets"."id"="AlbumEntity_AlbumEntity__AlbumEntity_assets"."assetsId" AND "AlbumEntity__AlbumEntity_assets"."deletedAt" IS NULL LEFT JOIN "albums_shared_users_users" "AlbumEntity_AlbumEntity__AlbumEntity_sharedUsers" ON "AlbumEntity_AlbumEntity__AlbumEntity_sharedUsers"."albumsId"="AlbumEntity"."id" LEFT JOIN "users" "AlbumEntity__AlbumEntity_sharedUsers" ON "AlbumEntity__AlbumEntity_sharedUsers"."id"="AlbumEntity_AlbumEntity__AlbumEntity_sharedUsers"."usersId" AND "AlbumEntity__AlbumEntity_sharedUsers"."deletedAt" IS NULL WHERE ( ("AlbumEntity"."ownerId" = $1 AND "AlbumEntity__AlbumEntity_assets"."id" = $2) OR ("AlbumEntity__AlbumEntity_sharedUsers"."id" = $3 AND "AlbumEntity__AlbumEntity_assets"."id" = $4) OR ("AlbumEntity"."ownerId" = $5 AND "AlbumEntity__AlbumEntity_assets"."livePhotoVideoId" = $6) OR ("AlbumEntity__AlbumEntity_sharedUsers"."id" = $7 AND "AlbumEntity__AlbumEntity_assets"."livePhotoVideoId" = $8) ) AND "AlbumEntity"."deletedAt" IS NULL ) LIMIT 1 -- After SELECT "asset"."id" AS "assetId", "asset"."livePhotoVideoId" AS "livePhotoVideoId" FROM "albums" "album" INNER JOIN "albums_assets_assets" "album_asset" ON "album_asset"."albumsId"="album"."id" INNER JOIN "assets" "asset" ON "asset"."id"="album_asset"."assetsId" AND "asset"."deletedAt" IS NULL LEFT JOIN "albums_shared_users_users" "album_sharedUsers" ON "album_sharedUsers"."albumsId"="album"."id" LEFT JOIN "users" "sharedUsers" ON "sharedUsers"."id"="album_sharedUsers"."usersId" AND "sharedUsers"."deletedAt" IS NULL WHERE ( "album"."ownerId" = $1 OR "sharedUsers"."id" = $2 ) AND ( "asset"."id" IN ($3, $4) OR "asset"."livePhotoVideoId" IN ($5, $6) ) AND "album"."deletedAt" IS NULL ``` * `asset` owner access: ```sql -- Before SELECT 1 AS "row_exists" FROM (SELECT 1 AS dummy_column) "dummy_table" WHERE EXISTS ( SELECT 1 FROM "assets" "AssetEntity" WHERE "AssetEntity"."id" = $1 AND "AssetEntity"."ownerId" = $2 ) LIMIT 1 -- After SELECT "AssetEntity"."id" AS "AssetEntity_id" FROM "assets" "AssetEntity" WHERE "AssetEntity"."id" IN ($1, $2) AND "AssetEntity"."ownerId" = $3 ``` * `asset` partner access: ```sql -- Before SELECT 1 AS "row_exists" FROM (SELECT 1 AS dummy_column) "dummy_table" WHERE EXISTS ( SELECT 1 FROM "partners" "PartnerEntity" LEFT JOIN "users" "PartnerEntity__PartnerEntity_sharedWith" ON "PartnerEntity__PartnerEntity_sharedWith"."id"="PartnerEntity"."sharedWithId" AND "PartnerEntity__PartnerEntity_sharedWith"."deletedAt" IS NULL LEFT JOIN "users" "PartnerEntity__PartnerEntity_sharedBy" ON "PartnerEntity__PartnerEntity_sharedBy"."id"="PartnerEntity"."sharedById" AND "PartnerEntity__PartnerEntity_sharedBy"."deletedAt" IS NULL LEFT JOIN "assets" "0aabe9f4a62b794e2c24a074297e534f51a4ac6c" ON "0aabe9f4a62b794e2c24a074297e534f51a4ac6c"."ownerId"="PartnerEntity__PartnerEntity_sharedBy"."id" AND "0aabe9f4a62b794e2c24a074297e534f51a4ac6c"."deletedAt" IS NULL LEFT JOIN "users" "PartnerEntity__sharedBy" ON "PartnerEntity__sharedBy"."id"="PartnerEntity"."sharedById" AND "PartnerEntity__sharedBy"."deletedAt" IS NULL LEFT JOIN "users" "PartnerEntity__sharedWith" ON "PartnerEntity__sharedWith"."id"="PartnerEntity"."sharedWithId" AND "PartnerEntity__sharedWith"."deletedAt" IS NULL WHERE "PartnerEntity__PartnerEntity_sharedWith"."id" = $1 AND "0aabe9f4a62b794e2c24a074297e534f51a4ac6c"."id" = $2 ) LIMIT 1 -- After SELECT "asset"."id" AS "assetId" FROM "partners" "partner" INNER JOIN "users" "sharedBy" ON "sharedBy"."id"="partner"."sharedById" AND "sharedBy"."deletedAt" IS NULL INNER JOIN "assets" "asset" ON "asset"."ownerId"="sharedBy"."id" AND "asset"."deletedAt" IS NULL WHERE "partner"."sharedWithId" = $1 AND "asset"."id" IN ($2, $3) ``` * `asset` shared link access: ```sql -- Before SELECT 1 AS "row_exists" FROM (SELECT 1 AS dummy_column) "dummy_table" WHERE EXISTS ( SELECT 1 FROM "shared_links" "SharedLinkEntity" LEFT JOIN "albums" "SharedLinkEntity__SharedLinkEntity_album" ON "SharedLinkEntity__SharedLinkEntity_album"."id"="SharedLinkEntity"."albumId" AND "SharedLinkEntity__SharedLinkEntity_album"."deletedAt" IS NULL LEFT JOIN "albums_assets_assets" "760f12c00d97bdcec1ce224d1e3bf449859942b6" ON "760f12c00d97bdcec1ce224d1e3bf449859942b6"."albumsId"="SharedLinkEntity__SharedLinkEntity_album"."id" LEFT JOIN "assets" "4a35f463ae8c5544ede95c4b6d9ce8c686b6bfe6" ON "4a35f463ae8c5544ede95c4b6d9ce8c686b6bfe6"."id"="760f12c00d97bdcec1ce224d1e3bf449859942b6"."assetsId" AND "4a35f463ae8c5544ede95c4b6d9ce8c686b6bfe6"."deletedAt" IS NULL LEFT JOIN "shared_link__asset" "SharedLinkEntity__SharedLinkEntity_assets_SharedLinkEntity" ON "SharedLinkEntity__SharedLinkEntity_assets_SharedLinkEntity"."sharedLinksId"="SharedLinkEntity"."id" LEFT JOIN "assets" "SharedLinkEntity__SharedLinkEntity_assets" ON "SharedLinkEntity__SharedLinkEntity_assets"."id"="SharedLinkEntity__SharedLinkEntity_assets_SharedLinkEntity"."assetsId" AND "SharedLinkEntity__SharedLinkEntity_assets"."deletedAt" IS NULL WHERE ( ("SharedLinkEntity"."id" = $1 AND "4a35f463ae8c5544ede95c4b6d9ce8c686b6bfe6"."id" = $2) OR ("SharedLinkEntity"."id" = $3 AND "SharedLinkEntity__SharedLinkEntity_assets"."id" = $4) OR ("SharedLinkEntity"."id" = $5 AND "4a35f463ae8c5544ede95c4b6d9ce8c686b6bfe6"."livePhotoVideoId" = $6) OR ("SharedLinkEntity"."id" = $7 AND "SharedLinkEntity__SharedLinkEntity_assets"."livePhotoVideoId" = $8) ) ) LIMIT 1 -- After SELECT "assets"."id" AS "assetId", "assets"."livePhotoVideoId" AS "assetLivePhotoVideoId", "albumAssets"."id" AS "albumAssetId", "albumAssets"."livePhotoVideoId" AS "albumAssetLivePhotoVideoId" FROM "shared_links" "sharedLink" LEFT JOIN "albums" "album" ON "album"."id"="sharedLink"."albumId" AND "album"."deletedAt" IS NULL LEFT JOIN "shared_link__asset" "assets_sharedLink" ON "assets_sharedLink"."sharedLinksId"="sharedLink"."id" LEFT JOIN "assets" "assets" ON "assets"."id"="assets_sharedLink"."assetsId" AND "assets"."deletedAt" IS NULL LEFT JOIN "albums_assets_assets" "album_albumAssets" ON "album_albumAssets"."albumsId"="album"."id" LEFT JOIN "assets" "albumAssets" ON "albumAssets"."id"="album_albumAssets"."assetsId" AND "albumAssets"."deletedAt" IS NULL WHERE "sharedLink"."id" = $1 AND ( "assets"."id" IN ($2, $3) OR "albumAssets"."id" IN ($4, $5) OR "assets"."livePhotoVideoId" IN ($6, $7) OR "albumAssets"."livePhotoVideoId" IN ($8, $9) ) ```
776 lines
31 KiB
TypeScript
776 lines
31 KiB
TypeScript
import { AssetFaceEntity, Colorspace, SystemConfigKey } from '@app/infra/entities';
|
|
import { BadRequestException, NotFoundException } from '@nestjs/common';
|
|
import {
|
|
IAccessRepositoryMock,
|
|
assetStub,
|
|
authStub,
|
|
faceStub,
|
|
newAccessRepositoryMock,
|
|
newAssetRepositoryMock,
|
|
newJobRepositoryMock,
|
|
newMachineLearningRepositoryMock,
|
|
newMediaRepositoryMock,
|
|
newMoveRepositoryMock,
|
|
newPersonRepositoryMock,
|
|
newSearchRepositoryMock,
|
|
newStorageRepositoryMock,
|
|
newSystemConfigRepositoryMock,
|
|
personStub,
|
|
} from '@test';
|
|
import { BulkIdErrorReason } from '../asset';
|
|
import { JobName } from '../job';
|
|
import {
|
|
IAssetRepository,
|
|
IJobRepository,
|
|
IMachineLearningRepository,
|
|
IMediaRepository,
|
|
IMoveRepository,
|
|
IPersonRepository,
|
|
ISearchRepository,
|
|
IStorageRepository,
|
|
ISystemConfigRepository,
|
|
WithoutProperty,
|
|
} from '../repositories';
|
|
import { PersonResponseDto } from './person.dto';
|
|
import { PersonService } from './person.service';
|
|
|
|
const responseDto: PersonResponseDto = {
|
|
id: 'person-1',
|
|
name: 'Person 1',
|
|
birthDate: null,
|
|
thumbnailPath: '/path/to/thumbnail.jpg',
|
|
isHidden: false,
|
|
};
|
|
|
|
const statistics = { assets: 3 };
|
|
|
|
const croppedFace = Buffer.from('Cropped Face');
|
|
|
|
const detectFaceMock = {
|
|
assetId: 'asset-1',
|
|
personId: 'person-1',
|
|
boundingBox: {
|
|
x1: 100,
|
|
y1: 100,
|
|
x2: 200,
|
|
y2: 200,
|
|
},
|
|
imageHeight: 500,
|
|
imageWidth: 400,
|
|
embedding: [1, 2, 3, 4],
|
|
score: 0.2,
|
|
};
|
|
|
|
const faceSearch = {
|
|
noMatch: {
|
|
total: 0,
|
|
count: 0,
|
|
page: 1,
|
|
items: [],
|
|
distances: [],
|
|
facets: [],
|
|
},
|
|
oneMatch: {
|
|
total: 1,
|
|
count: 1,
|
|
page: 1,
|
|
items: [faceStub.face1],
|
|
distances: [0.1],
|
|
facets: [],
|
|
},
|
|
oneRemoteMatch: {
|
|
total: 1,
|
|
count: 1,
|
|
page: 1,
|
|
items: [faceStub.face1],
|
|
distances: [0.8],
|
|
facets: [],
|
|
},
|
|
};
|
|
|
|
describe(PersonService.name, () => {
|
|
let accessMock: IAccessRepositoryMock;
|
|
let assetMock: jest.Mocked<IAssetRepository>;
|
|
let configMock: jest.Mocked<ISystemConfigRepository>;
|
|
let jobMock: jest.Mocked<IJobRepository>;
|
|
let machineLearningMock: jest.Mocked<IMachineLearningRepository>;
|
|
let mediaMock: jest.Mocked<IMediaRepository>;
|
|
let moveMock: jest.Mocked<IMoveRepository>;
|
|
let personMock: jest.Mocked<IPersonRepository>;
|
|
let searchMock: jest.Mocked<ISearchRepository>;
|
|
let storageMock: jest.Mocked<IStorageRepository>;
|
|
let sut: PersonService;
|
|
|
|
beforeEach(async () => {
|
|
accessMock = newAccessRepositoryMock();
|
|
assetMock = newAssetRepositoryMock();
|
|
configMock = newSystemConfigRepositoryMock();
|
|
jobMock = newJobRepositoryMock();
|
|
machineLearningMock = newMachineLearningRepositoryMock();
|
|
moveMock = newMoveRepositoryMock();
|
|
mediaMock = newMediaRepositoryMock();
|
|
personMock = newPersonRepositoryMock();
|
|
searchMock = newSearchRepositoryMock();
|
|
storageMock = newStorageRepositoryMock();
|
|
sut = new PersonService(
|
|
accessMock,
|
|
assetMock,
|
|
machineLearningMock,
|
|
moveMock,
|
|
mediaMock,
|
|
personMock,
|
|
searchMock,
|
|
configMock,
|
|
storageMock,
|
|
jobMock,
|
|
);
|
|
|
|
mediaMock.crop.mockResolvedValue(croppedFace);
|
|
});
|
|
|
|
it('should be defined', () => {
|
|
expect(sut).toBeDefined();
|
|
});
|
|
|
|
describe('getAll', () => {
|
|
it('should get all people with thumbnails', async () => {
|
|
personMock.getAllForUser.mockResolvedValue([personStub.withName, personStub.noThumbnail]);
|
|
await expect(sut.getAll(authStub.admin, { withHidden: undefined })).resolves.toEqual({
|
|
total: 1,
|
|
visible: 1,
|
|
people: [responseDto],
|
|
});
|
|
expect(personMock.getAllForUser).toHaveBeenCalledWith(authStub.admin.id, {
|
|
minimumFaceCount: 1,
|
|
withHidden: false,
|
|
});
|
|
});
|
|
it('should get all visible people with thumbnails', async () => {
|
|
personMock.getAllForUser.mockResolvedValue([personStub.withName, personStub.hidden]);
|
|
await expect(sut.getAll(authStub.admin, { withHidden: false })).resolves.toEqual({
|
|
total: 2,
|
|
visible: 1,
|
|
people: [responseDto],
|
|
});
|
|
expect(personMock.getAllForUser).toHaveBeenCalledWith(authStub.admin.id, {
|
|
minimumFaceCount: 1,
|
|
withHidden: false,
|
|
});
|
|
});
|
|
it('should get all hidden and visible people with thumbnails', async () => {
|
|
personMock.getAllForUser.mockResolvedValue([personStub.withName, personStub.hidden]);
|
|
await expect(sut.getAll(authStub.admin, { withHidden: true })).resolves.toEqual({
|
|
total: 2,
|
|
visible: 1,
|
|
people: [
|
|
responseDto,
|
|
{
|
|
id: 'person-1',
|
|
name: '',
|
|
birthDate: null,
|
|
thumbnailPath: '/path/to/thumbnail.jpg',
|
|
isHidden: true,
|
|
},
|
|
],
|
|
});
|
|
expect(personMock.getAllForUser).toHaveBeenCalledWith(authStub.admin.id, {
|
|
minimumFaceCount: 1,
|
|
withHidden: true,
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('getById', () => {
|
|
it('should require person.read permission', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.withName);
|
|
await expect(sut.getById(authStub.admin, 'person-1')).rejects.toBeInstanceOf(BadRequestException);
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
|
|
it('should throw a bad request when person is not found', async () => {
|
|
personMock.getById.mockResolvedValue(null);
|
|
accessMock.person.checkOwnerAccess.mockResolvedValue(new Set(['person-1']));
|
|
await expect(sut.getById(authStub.admin, 'person-1')).rejects.toBeInstanceOf(BadRequestException);
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
|
|
it('should get a person by id', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.withName);
|
|
accessMock.person.checkOwnerAccess.mockResolvedValue(new Set(['person-1']));
|
|
await expect(sut.getById(authStub.admin, 'person-1')).resolves.toEqual(responseDto);
|
|
expect(personMock.getById).toHaveBeenCalledWith('person-1');
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
});
|
|
|
|
describe('getThumbnail', () => {
|
|
it('should require person.read permission', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.noName);
|
|
await expect(sut.getThumbnail(authStub.admin, 'person-1')).rejects.toBeInstanceOf(BadRequestException);
|
|
expect(storageMock.createReadStream).not.toHaveBeenCalled();
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
|
|
it('should throw an error when personId is invalid', async () => {
|
|
personMock.getById.mockResolvedValue(null);
|
|
accessMock.person.checkOwnerAccess.mockResolvedValue(new Set(['person-1']));
|
|
await expect(sut.getThumbnail(authStub.admin, 'person-1')).rejects.toBeInstanceOf(NotFoundException);
|
|
expect(storageMock.createReadStream).not.toHaveBeenCalled();
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
|
|
it('should throw an error when person has no thumbnail', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.noThumbnail);
|
|
accessMock.person.checkOwnerAccess.mockResolvedValue(new Set(['person-1']));
|
|
await expect(sut.getThumbnail(authStub.admin, 'person-1')).rejects.toBeInstanceOf(NotFoundException);
|
|
expect(storageMock.createReadStream).not.toHaveBeenCalled();
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
|
|
it('should serve the thumbnail', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.noName);
|
|
accessMock.person.checkOwnerAccess.mockResolvedValue(new Set(['person-1']));
|
|
await sut.getThumbnail(authStub.admin, 'person-1');
|
|
expect(storageMock.createReadStream).toHaveBeenCalledWith('/path/to/thumbnail.jpg', 'image/jpeg');
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
});
|
|
|
|
describe('getAssets', () => {
|
|
it('should require person.read permission', async () => {
|
|
personMock.getAssets.mockResolvedValue([assetStub.image, assetStub.video]);
|
|
await expect(sut.getAssets(authStub.admin, 'person-1')).rejects.toBeInstanceOf(BadRequestException);
|
|
expect(personMock.getAssets).not.toHaveBeenCalled();
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
|
|
it("should return a person's assets", async () => {
|
|
personMock.getAssets.mockResolvedValue([assetStub.image, assetStub.video]);
|
|
accessMock.person.checkOwnerAccess.mockResolvedValue(new Set(['person-1']));
|
|
await sut.getAssets(authStub.admin, 'person-1');
|
|
expect(personMock.getAssets).toHaveBeenCalledWith('person-1');
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
});
|
|
|
|
describe('update', () => {
|
|
it('should require person.write permission', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.noName);
|
|
await expect(sut.update(authStub.admin, 'person-1', { name: 'Person 1' })).rejects.toBeInstanceOf(
|
|
BadRequestException,
|
|
);
|
|
expect(personMock.update).not.toHaveBeenCalled();
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
|
|
it('should throw an error when personId is invalid', async () => {
|
|
personMock.getById.mockResolvedValue(null);
|
|
accessMock.person.checkOwnerAccess.mockResolvedValue(new Set(['person-1']));
|
|
await expect(sut.update(authStub.admin, 'person-1', { name: 'Person 1' })).rejects.toBeInstanceOf(
|
|
BadRequestException,
|
|
);
|
|
expect(personMock.update).not.toHaveBeenCalled();
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
|
|
it("should update a person's name", async () => {
|
|
personMock.getById.mockResolvedValue(personStub.noName);
|
|
personMock.update.mockResolvedValue(personStub.withName);
|
|
personMock.getAssets.mockResolvedValue([assetStub.image]);
|
|
accessMock.person.checkOwnerAccess.mockResolvedValue(new Set(['person-1']));
|
|
|
|
await expect(sut.update(authStub.admin, 'person-1', { name: 'Person 1' })).resolves.toEqual(responseDto);
|
|
|
|
expect(personMock.getById).toHaveBeenCalledWith('person-1');
|
|
expect(personMock.update).toHaveBeenCalledWith({ id: 'person-1', name: 'Person 1' });
|
|
expect(jobMock.queue).toHaveBeenCalledWith({
|
|
name: JobName.SEARCH_INDEX_ASSET,
|
|
data: { ids: [assetStub.image.id] },
|
|
});
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
|
|
it("should update a person's date of birth", async () => {
|
|
personMock.getById.mockResolvedValue(personStub.noBirthDate);
|
|
personMock.update.mockResolvedValue(personStub.withBirthDate);
|
|
personMock.getAssets.mockResolvedValue([assetStub.image]);
|
|
accessMock.person.checkOwnerAccess.mockResolvedValue(new Set(['person-1']));
|
|
|
|
await expect(sut.update(authStub.admin, 'person-1', { birthDate: new Date('1976-06-30') })).resolves.toEqual({
|
|
id: 'person-1',
|
|
name: 'Person 1',
|
|
birthDate: new Date('1976-06-30'),
|
|
thumbnailPath: '/path/to/thumbnail.jpg',
|
|
isHidden: false,
|
|
});
|
|
|
|
expect(personMock.getById).toHaveBeenCalledWith('person-1');
|
|
expect(personMock.update).toHaveBeenCalledWith({ id: 'person-1', birthDate: new Date('1976-06-30') });
|
|
expect(jobMock.queue).not.toHaveBeenCalled();
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
|
|
it('should update a person visibility', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.hidden);
|
|
personMock.update.mockResolvedValue(personStub.withName);
|
|
personMock.getAssets.mockResolvedValue([assetStub.image]);
|
|
accessMock.person.checkOwnerAccess.mockResolvedValue(new Set(['person-1']));
|
|
|
|
await expect(sut.update(authStub.admin, 'person-1', { isHidden: false })).resolves.toEqual(responseDto);
|
|
|
|
expect(personMock.getById).toHaveBeenCalledWith('person-1');
|
|
expect(personMock.update).toHaveBeenCalledWith({ id: 'person-1', isHidden: false });
|
|
expect(jobMock.queue).toHaveBeenCalledWith({
|
|
name: JobName.SEARCH_INDEX_ASSET,
|
|
data: { ids: [assetStub.image.id] },
|
|
});
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
|
|
it("should update a person's thumbnailPath", async () => {
|
|
personMock.getById.mockResolvedValue(personStub.withName);
|
|
personMock.update.mockResolvedValue(personStub.withName);
|
|
personMock.getFacesByIds.mockResolvedValue([faceStub.face1]);
|
|
accessMock.asset.checkOwnerAccess.mockResolvedValue(new Set([assetStub.image.id]));
|
|
accessMock.person.checkOwnerAccess.mockResolvedValue(new Set(['person-1']));
|
|
|
|
await expect(
|
|
sut.update(authStub.admin, 'person-1', { featureFaceAssetId: faceStub.face1.assetId }),
|
|
).resolves.toEqual(responseDto);
|
|
|
|
expect(personMock.getById).toHaveBeenCalledWith('person-1');
|
|
expect(personMock.update).toHaveBeenCalledWith({ id: 'person-1', faceAssetId: faceStub.face1.assetId });
|
|
expect(personMock.getFacesByIds).toHaveBeenCalledWith([
|
|
{
|
|
assetId: faceStub.face1.assetId,
|
|
personId: 'person-1',
|
|
},
|
|
]);
|
|
expect(jobMock.queue).toHaveBeenCalledWith({ name: JobName.GENERATE_PERSON_THUMBNAIL, data: { id: 'person-1' } });
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
|
|
it('should throw an error when the face feature assetId is invalid', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.withName);
|
|
accessMock.person.checkOwnerAccess.mockResolvedValue(new Set(['person-1']));
|
|
|
|
await expect(sut.update(authStub.admin, 'person-1', { featureFaceAssetId: '-1' })).rejects.toThrow(
|
|
BadRequestException,
|
|
);
|
|
expect(personMock.update).not.toHaveBeenCalled();
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
});
|
|
|
|
describe('updateAll', () => {
|
|
it('should throw an error when personId is invalid', async () => {
|
|
personMock.getById.mockResolvedValue(null);
|
|
accessMock.person.checkOwnerAccess.mockResolvedValue(new Set(['person-1']));
|
|
|
|
await expect(
|
|
sut.updatePeople(authStub.admin, { people: [{ id: 'person-1', name: 'Person 1' }] }),
|
|
).resolves.toEqual([{ error: BulkIdErrorReason.UNKNOWN, id: 'person-1', success: false }]);
|
|
expect(personMock.update).not.toHaveBeenCalled();
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
});
|
|
|
|
describe('handlePersonCleanup', () => {
|
|
it('should delete people without faces', async () => {
|
|
personMock.getAllWithoutFaces.mockResolvedValue([personStub.noName]);
|
|
|
|
await sut.handlePersonCleanup();
|
|
|
|
expect(jobMock.queue).toHaveBeenCalledWith({ name: JobName.PERSON_DELETE, data: { id: personStub.noName.id } });
|
|
});
|
|
});
|
|
|
|
describe('handleQueueRecognizeFaces', () => {
|
|
it('should return if machine learning is disabled', async () => {
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.MACHINE_LEARNING_ENABLED, value: false }]);
|
|
|
|
await expect(sut.handleQueueRecognizeFaces({})).resolves.toBe(true);
|
|
expect(jobMock.queue).not.toHaveBeenCalled();
|
|
expect(configMock.load).toHaveBeenCalled();
|
|
});
|
|
|
|
it('should queue missing assets', async () => {
|
|
assetMock.getWithout.mockResolvedValue({
|
|
items: [assetStub.image],
|
|
hasNextPage: false,
|
|
});
|
|
await sut.handleQueueRecognizeFaces({});
|
|
|
|
expect(assetMock.getWithout).toHaveBeenCalledWith({ skip: 0, take: 1000 }, WithoutProperty.FACES);
|
|
expect(jobMock.queue).toHaveBeenCalledWith({
|
|
name: JobName.RECOGNIZE_FACES,
|
|
data: { id: assetStub.image.id },
|
|
});
|
|
});
|
|
|
|
it('should queue all assets', async () => {
|
|
assetMock.getAll.mockResolvedValue({
|
|
items: [assetStub.image],
|
|
hasNextPage: false,
|
|
});
|
|
personMock.getAll.mockResolvedValue([personStub.withName]);
|
|
searchMock.deleteAllFaces.mockResolvedValue(100);
|
|
|
|
await sut.handleQueueRecognizeFaces({ force: true });
|
|
|
|
expect(assetMock.getAll).toHaveBeenCalled();
|
|
expect(jobMock.queue).toHaveBeenCalledWith({
|
|
name: JobName.RECOGNIZE_FACES,
|
|
data: { id: assetStub.image.id },
|
|
});
|
|
expect(jobMock.queue).toHaveBeenCalledWith({
|
|
name: JobName.PERSON_DELETE,
|
|
data: { id: personStub.withName.id },
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('handleRecognizeFaces', () => {
|
|
it('should return if machine learning is disabled', async () => {
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.MACHINE_LEARNING_ENABLED, value: false }]);
|
|
|
|
await expect(sut.handleRecognizeFaces({ id: 'foo' })).resolves.toBe(true);
|
|
expect(assetMock.getByIds).not.toHaveBeenCalled();
|
|
expect(configMock.load).toHaveBeenCalled();
|
|
});
|
|
|
|
it('should skip when no resize path', async () => {
|
|
assetMock.getByIds.mockResolvedValue([assetStub.noResizePath]);
|
|
await sut.handleRecognizeFaces({ id: assetStub.noResizePath.id });
|
|
expect(machineLearningMock.detectFaces).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it('should skip it the asset has already been processed', async () => {
|
|
assetMock.getByIds.mockResolvedValue([
|
|
{
|
|
...assetStub.noResizePath,
|
|
faces: [
|
|
{
|
|
id: 'asset-face-1',
|
|
assetId: assetStub.noResizePath.id,
|
|
personId: faceStub.face1.personId,
|
|
} as AssetFaceEntity,
|
|
],
|
|
},
|
|
]);
|
|
await sut.handleRecognizeFaces({ id: assetStub.noResizePath.id });
|
|
expect(machineLearningMock.detectFaces).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it('should handle no results', async () => {
|
|
const start = Date.now();
|
|
|
|
machineLearningMock.detectFaces.mockResolvedValue([]);
|
|
assetMock.getByIds.mockResolvedValue([assetStub.image]);
|
|
await sut.handleRecognizeFaces({ id: assetStub.image.id });
|
|
expect(machineLearningMock.detectFaces).toHaveBeenCalledWith(
|
|
'http://immich-machine-learning:3003',
|
|
{
|
|
imagePath: assetStub.image.resizePath,
|
|
},
|
|
{
|
|
enabled: true,
|
|
maxDistance: 0.6,
|
|
minScore: 0.7,
|
|
minFaces: 1,
|
|
modelName: 'buffalo_l',
|
|
},
|
|
);
|
|
expect(personMock.createFace).not.toHaveBeenCalled();
|
|
expect(jobMock.queue).not.toHaveBeenCalled();
|
|
|
|
expect(assetMock.upsertJobStatus).toHaveBeenCalledWith({
|
|
assetId: assetStub.image.id,
|
|
facesRecognizedAt: expect.any(Date),
|
|
});
|
|
expect(assetMock.upsertJobStatus.mock.calls[0][0].facesRecognizedAt?.getTime()).toBeGreaterThan(start);
|
|
});
|
|
|
|
it('should match existing people', async () => {
|
|
machineLearningMock.detectFaces.mockResolvedValue([detectFaceMock]);
|
|
searchMock.searchFaces.mockResolvedValue(faceSearch.oneMatch);
|
|
assetMock.getByIds.mockResolvedValue([assetStub.image]);
|
|
await sut.handleRecognizeFaces({ id: assetStub.image.id });
|
|
|
|
expect(personMock.createFace).toHaveBeenCalledWith({
|
|
personId: 'person-1',
|
|
assetId: 'asset-id',
|
|
embedding: [1, 2, 3, 4],
|
|
boundingBoxX1: 100,
|
|
boundingBoxY1: 100,
|
|
boundingBoxX2: 200,
|
|
boundingBoxY2: 200,
|
|
imageHeight: 500,
|
|
imageWidth: 400,
|
|
});
|
|
});
|
|
|
|
it('should create a new person', async () => {
|
|
machineLearningMock.detectFaces.mockResolvedValue([detectFaceMock]);
|
|
searchMock.searchFaces.mockResolvedValue(faceSearch.oneRemoteMatch);
|
|
personMock.create.mockResolvedValue(personStub.noName);
|
|
assetMock.getByIds.mockResolvedValue([assetStub.image]);
|
|
|
|
await sut.handleRecognizeFaces({ id: assetStub.image.id });
|
|
|
|
expect(personMock.create).toHaveBeenCalledWith({ ownerId: assetStub.image.ownerId });
|
|
expect(personMock.createFace).toHaveBeenCalledWith({
|
|
personId: 'person-1',
|
|
assetId: 'asset-id',
|
|
embedding: [1, 2, 3, 4],
|
|
boundingBoxX1: 100,
|
|
boundingBoxY1: 100,
|
|
boundingBoxX2: 200,
|
|
boundingBoxY2: 200,
|
|
imageHeight: 500,
|
|
imageWidth: 400,
|
|
});
|
|
expect(jobMock.queue.mock.calls).toEqual([
|
|
[{ name: JobName.SEARCH_INDEX_FACE, data: { personId: 'person-1', assetId: 'asset-id' } }],
|
|
[{ name: JobName.GENERATE_PERSON_THUMBNAIL, data: { id: 'person-1' } }],
|
|
]);
|
|
});
|
|
});
|
|
describe('handleGeneratePersonThumbnail', () => {
|
|
it('should return if machine learning is disabled', async () => {
|
|
configMock.load.mockResolvedValue([{ key: SystemConfigKey.MACHINE_LEARNING_ENABLED, value: false }]);
|
|
|
|
await expect(sut.handleGeneratePersonThumbnail({ id: 'person-1' })).resolves.toBe(true);
|
|
expect(assetMock.getByIds).not.toHaveBeenCalled();
|
|
expect(configMock.load).toHaveBeenCalled();
|
|
});
|
|
|
|
it('should skip a person not found', async () => {
|
|
personMock.getById.mockResolvedValue(null);
|
|
await sut.handleGeneratePersonThumbnail({ id: 'person-1' });
|
|
expect(mediaMock.crop).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it('should skip a person without a face asset id', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.noThumbnail);
|
|
await sut.handleGeneratePersonThumbnail({ id: 'person-1' });
|
|
expect(mediaMock.crop).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it('should skip an person with a face asset id not found', async () => {
|
|
personMock.getById.mockResolvedValue({ ...personStub.primaryPerson, faceAssetId: faceStub.middle.assetId });
|
|
personMock.getFacesByIds.mockResolvedValue([faceStub.face1]);
|
|
await sut.handleGeneratePersonThumbnail({ id: 'person-1' });
|
|
expect(mediaMock.crop).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it('should skip a person with a face asset id without a thumbnail', async () => {
|
|
personMock.getById.mockResolvedValue({ ...personStub.primaryPerson, faceAssetId: faceStub.middle.assetId });
|
|
personMock.getFacesByIds.mockResolvedValue([faceStub.face1]);
|
|
assetMock.getByIds.mockResolvedValue([assetStub.noResizePath]);
|
|
await sut.handleGeneratePersonThumbnail({ id: 'person-1' });
|
|
expect(mediaMock.crop).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it('should generate a thumbnail', async () => {
|
|
personMock.getById.mockResolvedValue({ ...personStub.primaryPerson, faceAssetId: faceStub.middle.assetId });
|
|
personMock.getFacesByIds.mockResolvedValue([faceStub.middle]);
|
|
assetMock.getByIds.mockResolvedValue([assetStub.primaryImage]);
|
|
|
|
await sut.handleGeneratePersonThumbnail({ id: 'person-1' });
|
|
|
|
expect(assetMock.getByIds).toHaveBeenCalledWith([faceStub.middle.assetId]);
|
|
expect(storageMock.mkdirSync).toHaveBeenCalledWith('upload/thumbs/admin_id/pe/rs');
|
|
expect(mediaMock.crop).toHaveBeenCalledWith('/uploads/admin-id/thumbs/path.jpg', {
|
|
left: 95,
|
|
top: 95,
|
|
width: 110,
|
|
height: 110,
|
|
});
|
|
expect(mediaMock.resize).toHaveBeenCalledWith(croppedFace, 'upload/thumbs/admin_id/pe/rs/person-1.jpeg', {
|
|
format: 'jpeg',
|
|
size: 250,
|
|
quality: 80,
|
|
colorspace: Colorspace.P3,
|
|
});
|
|
expect(personMock.update).toHaveBeenCalledWith({
|
|
id: 'person-1',
|
|
thumbnailPath: 'upload/thumbs/admin_id/pe/rs/person-1.jpeg',
|
|
});
|
|
});
|
|
|
|
it('should generate a thumbnail without going negative', async () => {
|
|
personMock.getById.mockResolvedValue({ ...personStub.primaryPerson, faceAssetId: faceStub.start.assetId });
|
|
personMock.getFacesByIds.mockResolvedValue([faceStub.start]);
|
|
assetMock.getByIds.mockResolvedValue([assetStub.image]);
|
|
|
|
await sut.handleGeneratePersonThumbnail({ id: 'person-1' });
|
|
|
|
expect(mediaMock.crop).toHaveBeenCalledWith('/uploads/user-id/thumbs/path.jpg', {
|
|
left: 0,
|
|
top: 0,
|
|
width: 510,
|
|
height: 510,
|
|
});
|
|
expect(mediaMock.resize).toHaveBeenCalledWith(croppedFace, 'upload/thumbs/admin_id/pe/rs/person-1.jpeg', {
|
|
format: 'jpeg',
|
|
size: 250,
|
|
quality: 80,
|
|
colorspace: Colorspace.P3,
|
|
});
|
|
});
|
|
|
|
it('should generate a thumbnail without overflowing', async () => {
|
|
personMock.getById.mockResolvedValue({ ...personStub.primaryPerson, faceAssetId: faceStub.end.assetId });
|
|
personMock.getFacesByIds.mockResolvedValue([faceStub.end]);
|
|
assetMock.getByIds.mockResolvedValue([assetStub.primaryImage]);
|
|
|
|
await sut.handleGeneratePersonThumbnail({ id: 'person-1' });
|
|
|
|
expect(mediaMock.crop).toHaveBeenCalledWith('/uploads/admin-id/thumbs/path.jpg', {
|
|
left: 297,
|
|
top: 297,
|
|
width: 202,
|
|
height: 202,
|
|
});
|
|
expect(mediaMock.resize).toHaveBeenCalledWith(croppedFace, 'upload/thumbs/admin_id/pe/rs/person-1.jpeg', {
|
|
format: 'jpeg',
|
|
size: 250,
|
|
quality: 80,
|
|
colorspace: Colorspace.P3,
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('mergePerson', () => {
|
|
it('should require person.write and person.merge permission', async () => {
|
|
personMock.getById.mockResolvedValueOnce(personStub.primaryPerson);
|
|
personMock.getById.mockResolvedValueOnce(personStub.mergePerson);
|
|
personMock.prepareReassignFaces.mockResolvedValue([]);
|
|
personMock.delete.mockResolvedValue(personStub.mergePerson);
|
|
|
|
await expect(sut.mergePerson(authStub.admin, 'person-1', { ids: ['person-2'] })).rejects.toBeInstanceOf(
|
|
BadRequestException,
|
|
);
|
|
|
|
expect(personMock.prepareReassignFaces).not.toHaveBeenCalled();
|
|
|
|
expect(personMock.reassignFaces).not.toHaveBeenCalled();
|
|
|
|
expect(personMock.delete).not.toHaveBeenCalled();
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
|
|
it('should merge two people', async () => {
|
|
personMock.getById.mockResolvedValueOnce(personStub.primaryPerson);
|
|
personMock.getById.mockResolvedValueOnce(personStub.mergePerson);
|
|
personMock.prepareReassignFaces.mockResolvedValue([]);
|
|
personMock.delete.mockResolvedValue(personStub.mergePerson);
|
|
accessMock.person.checkOwnerAccess.mockResolvedValueOnce(new Set(['person-1']));
|
|
accessMock.person.checkOwnerAccess.mockResolvedValueOnce(new Set(['person-2']));
|
|
|
|
await expect(sut.mergePerson(authStub.admin, 'person-1', { ids: ['person-2'] })).resolves.toEqual([
|
|
{ id: 'person-2', success: true },
|
|
]);
|
|
|
|
expect(personMock.prepareReassignFaces).toHaveBeenCalledWith({
|
|
newPersonId: personStub.primaryPerson.id,
|
|
oldPersonId: personStub.mergePerson.id,
|
|
});
|
|
|
|
expect(personMock.reassignFaces).toHaveBeenCalledWith({
|
|
newPersonId: personStub.primaryPerson.id,
|
|
oldPersonId: personStub.mergePerson.id,
|
|
});
|
|
|
|
expect(jobMock.queue).toHaveBeenCalledWith({
|
|
name: JobName.PERSON_DELETE,
|
|
data: { id: personStub.mergePerson.id },
|
|
});
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
|
|
it('should delete conflicting faces before merging', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.primaryPerson);
|
|
personMock.getById.mockResolvedValue(personStub.mergePerson);
|
|
personMock.prepareReassignFaces.mockResolvedValue([assetStub.image.id]);
|
|
accessMock.person.checkOwnerAccess.mockResolvedValueOnce(new Set(['person-1']));
|
|
accessMock.person.checkOwnerAccess.mockResolvedValueOnce(new Set(['person-2']));
|
|
|
|
await expect(sut.mergePerson(authStub.admin, 'person-1', { ids: ['person-2'] })).resolves.toEqual([
|
|
{ id: 'person-2', success: true },
|
|
]);
|
|
|
|
expect(personMock.prepareReassignFaces).toHaveBeenCalledWith({
|
|
newPersonId: personStub.primaryPerson.id,
|
|
oldPersonId: personStub.mergePerson.id,
|
|
});
|
|
|
|
expect(jobMock.queue).toHaveBeenCalledWith({
|
|
name: JobName.SEARCH_REMOVE_FACE,
|
|
data: { assetId: assetStub.image.id, personId: personStub.mergePerson.id },
|
|
});
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
|
|
it('should throw an error when the primary person is not found', async () => {
|
|
personMock.getById.mockResolvedValue(null);
|
|
accessMock.person.checkOwnerAccess.mockResolvedValue(new Set(['person-1']));
|
|
|
|
await expect(sut.mergePerson(authStub.admin, 'person-1', { ids: ['person-2'] })).rejects.toBeInstanceOf(
|
|
BadRequestException,
|
|
);
|
|
|
|
expect(personMock.delete).not.toHaveBeenCalled();
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
|
|
it('should handle invalid merge ids', async () => {
|
|
personMock.getById.mockResolvedValueOnce(personStub.primaryPerson);
|
|
personMock.getById.mockResolvedValueOnce(null);
|
|
accessMock.person.checkOwnerAccess.mockResolvedValueOnce(new Set(['person-1']));
|
|
accessMock.person.checkOwnerAccess.mockResolvedValueOnce(new Set(['person-2']));
|
|
|
|
await expect(sut.mergePerson(authStub.admin, 'person-1', { ids: ['person-2'] })).resolves.toEqual([
|
|
{ id: 'person-2', success: false, error: BulkIdErrorReason.NOT_FOUND },
|
|
]);
|
|
|
|
expect(personMock.prepareReassignFaces).not.toHaveBeenCalled();
|
|
expect(personMock.reassignFaces).not.toHaveBeenCalled();
|
|
expect(personMock.delete).not.toHaveBeenCalled();
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
|
|
it('should handle an error reassigning faces', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.primaryPerson);
|
|
personMock.getById.mockResolvedValue(personStub.mergePerson);
|
|
personMock.prepareReassignFaces.mockResolvedValue([assetStub.image.id]);
|
|
personMock.reassignFaces.mockRejectedValue(new Error('update failed'));
|
|
accessMock.person.checkOwnerAccess.mockResolvedValueOnce(new Set(['person-1']));
|
|
accessMock.person.checkOwnerAccess.mockResolvedValueOnce(new Set(['person-2']));
|
|
|
|
await expect(sut.mergePerson(authStub.admin, 'person-1', { ids: ['person-2'] })).resolves.toEqual([
|
|
{ id: 'person-2', success: false, error: BulkIdErrorReason.UNKNOWN },
|
|
]);
|
|
|
|
expect(personMock.delete).not.toHaveBeenCalled();
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
});
|
|
|
|
describe('getStatistics', () => {
|
|
it('should get correct number of person', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.primaryPerson);
|
|
personMock.getStatistics.mockResolvedValue(statistics);
|
|
accessMock.person.checkOwnerAccess.mockResolvedValue(new Set(['person-1']));
|
|
await expect(sut.getStatistics(authStub.admin, 'person-1')).resolves.toEqual({ assets: 3 });
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
|
|
it('should require person.read permission', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.primaryPerson);
|
|
await expect(sut.getStatistics(authStub.admin, 'person-1')).rejects.toBeInstanceOf(BadRequestException);
|
|
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.id, new Set(['person-1']));
|
|
});
|
|
});
|
|
});
|