Skip to content

Commit 33923a0

Browse files
committed
fix: album start/end dates on shared links
1 parent 63088b2 commit 33923a0

File tree

3 files changed

+113
-2
lines changed

3 files changed

+113
-2
lines changed

server/src/repositories/shared-link.repository.ts

Lines changed: 10 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -86,7 +86,16 @@ export class SharedLinkRepository {
8686
(join) => join.onTrue(),
8787
)
8888
.select((eb) =>
89-
eb.fn.coalesce(eb.fn.jsonAgg('assets').filterWhere('assets.id', 'is not', null), sql`'[]'`).as('assets'),
89+
eb.fn
90+
.coalesce(
91+
eb.fn
92+
.jsonAgg('assets')
93+
.orderBy('assets.fileCreatedAt', 'asc')
94+
.filterWhere('assets.id', 'is not', null),
95+
96+
sql`'[]'`,
97+
)
98+
.as('assets'),
9099
)
91100
.select((eb) => eb.fn.toJson('owner').as('owner'))
92101
.groupBy(['album.id', sql`"owner".*`])

server/test/medium.factory.ts

Lines changed: 3 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,7 @@ import { PartnerRepository } from 'src/repositories/partner.repository';
3333
import { PersonRepository } from 'src/repositories/person.repository';
3434
import { SearchRepository } from 'src/repositories/search.repository';
3535
import { SessionRepository } from 'src/repositories/session.repository';
36+
import { SharedLinkRepository } from 'src/repositories/shared-link.repository';
3637
import { StackRepository } from 'src/repositories/stack.repository';
3738
import { StorageRepository } from 'src/repositories/storage.repository';
3839
import { SyncCheckpointRepository } from 'src/repositories/sync-checkpoint.repository';
@@ -286,6 +287,7 @@ const newRealRepository = <T>(key: ClassConstructor<T>, db: Kysely<DB>): T => {
286287
case PersonRepository:
287288
case SearchRepository:
288289
case SessionRepository:
290+
case SharedLinkRepository:
289291
case StackRepository:
290292
case SyncRepository:
291293
case SyncCheckpointRepository:
@@ -391,7 +393,7 @@ const assetInsert = (asset: Partial<Insertable<AssetTable>> = {}) => {
391393
checksum: randomBytes(32),
392394
type: AssetType.Image,
393395
originalPath: '/path/to/something.jpg',
394-
ownerId: '@immich.cloud',
396+
ownerId: 'not-a-valid-uuid',
395397
isFavorite: false,
396398
fileCreatedAt: now,
397399
fileModifiedAt: now,
Lines changed: 100 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,100 @@
1+
import { Kysely } from 'kysely';
2+
import { randomBytes } from 'node:crypto';
3+
import { SharedLinkType } from 'src/enum';
4+
import { AccessRepository } from 'src/repositories/access.repository';
5+
import { DatabaseRepository } from 'src/repositories/database.repository';
6+
import { LoggingRepository } from 'src/repositories/logging.repository';
7+
import { SharedLinkRepository } from 'src/repositories/shared-link.repository';
8+
import { StorageRepository } from 'src/repositories/storage.repository';
9+
import { DB } from 'src/schema';
10+
import { SharedLinkService } from 'src/services/shared-link.service';
11+
import { newMediumService } from 'test/medium.factory';
12+
import { factory } from 'test/small.factory';
13+
import { getKyselyDB } from 'test/utils';
14+
15+
let defaultDatabase: Kysely<DB>;
16+
17+
const setup = (db?: Kysely<DB>) => {
18+
return newMediumService(SharedLinkService, {
19+
database: db || defaultDatabase,
20+
real: [AccessRepository, DatabaseRepository, SharedLinkRepository],
21+
mock: [LoggingRepository, StorageRepository],
22+
});
23+
};
24+
25+
beforeAll(async () => {
26+
defaultDatabase = await getKyselyDB();
27+
});
28+
29+
describe(SharedLinkService.name, () => {
30+
describe('get', () => {
31+
it('should return the correct dates on the shared link album', async () => {
32+
const { sut, ctx } = setup();
33+
34+
const { user } = await ctx.newUser();
35+
const auth = factory.auth({ user });
36+
const { album } = await ctx.newAlbum({ ownerId: user.id });
37+
38+
const dates = ['2021-01-01T00:00:00.000Z', '2022-01-01T00:00:00.000Z', '2020-01-01T00:00:00.000Z'];
39+
40+
for (const date of dates) {
41+
const { asset } = await ctx.newAsset({ fileCreatedAt: date, localDateTime: date, ownerId: user.id });
42+
await ctx.newExif({ assetId: asset.id, make: 'Canon' });
43+
await ctx.newAlbumAsset({ albumId: album.id, assetId: asset.id });
44+
}
45+
46+
const sharedLinkRepo = ctx.get(SharedLinkRepository);
47+
48+
const sharedLink = await sharedLinkRepo.create({
49+
key: randomBytes(16),
50+
id: factory.uuid(),
51+
userId: user.id,
52+
albumId: album.id,
53+
allowUpload: true,
54+
type: SharedLinkType.Album,
55+
});
56+
57+
await expect(sut.get(auth, sharedLink.id)).resolves.toMatchObject({
58+
album: expect.objectContaining({
59+
startDate: '2020-01-01T00:00:00+00:00',
60+
endDate: '2022-01-01T00:00:00+00:00',
61+
}),
62+
});
63+
});
64+
65+
it('should return the correct dates on the shared link album', async () => {
66+
const { sut, ctx } = setup();
67+
68+
const { user } = await ctx.newUser();
69+
const auth = factory.auth({ user });
70+
const { album } = await ctx.newAlbum({ ownerId: user.id });
71+
72+
const dates = ['2021-01-01T00:00:00.000Z', '2022-01-01T00:00:00.000Z', '2020-01-01T00:00:00.000Z'];
73+
74+
for (const date of dates) {
75+
const { asset } = await ctx.newAsset({ fileCreatedAt: date, localDateTime: date, ownerId: user.id });
76+
await ctx.newExif({ assetId: asset.id, make: 'Canon' });
77+
await ctx.newAlbumAsset({ albumId: album.id, assetId: asset.id });
78+
}
79+
80+
const sharedLinkRepo = ctx.get(SharedLinkRepository);
81+
82+
const sharedLink = await sharedLinkRepo.create({
83+
key: randomBytes(16),
84+
id: factory.uuid(),
85+
userId: user.id,
86+
albumId: album.id,
87+
allowUpload: true,
88+
type: SharedLinkType.Album,
89+
});
90+
91+
await expect(sut.get(auth, sharedLink.id)).resolves.toMatchObject({
92+
album: expect.objectContaining({
93+
startDate: '2020-01-01T00:00:00+00:00',
94+
endDate: '2022-01-01T00:00:00+00:00',
95+
}),
96+
//
97+
});
98+
});
99+
});
100+
});

0 commit comments

Comments
 (0)